LightOJ 1140: How Many Zeroes? (数位DP)
当前数位DP还不理解的点:
1:出口用i==0的方式
2:如何省略状态d(就是枚举下一个数的那个状态。当然枚举还是要的,怎么把空间省了)
总结:
1:此类DP,考虑转移的时候,应当同时考虑查询时候的情况。
2:考虑x在第i位之后,能遍历多少数字,其答案为(x%10i-1+1)
3:这里的记忆化搜索不太一样喔,出口一定要写在递归里,不然,查询状态下差到出口就会出错了~
类型:
数位DP
题意:
求[A,B]区间内的所有数,写下来之后,0的个数。(a,b 为 unsigned int)
思路:
我的笨拙暴力状态:
dp[i][d][okPre] 表示d开头的i位数,(okPre表示计算前导0的情况下,反之~),的0的个数。
那么。
dp[i][d][含] = dp[i-1][0~9(num[i-1])][含] + 10i-1(x%10i-1+1) * (d==0);
dp[i][d][不含] = dp[i-1][1~9(num[i-1])][含] + dp[i-1][0][d==0?不含:含] ;
出口:
dp[1][1~9][含] =dp[1][1~9][不含] = 0;
dp[1][0][不含] = dp[1][0][含] = 1;
当时确定出口的时候,不含的0应该是0还是1呢?不好确定,感觉是1,最后是通过试验确定的。
还是没有理解别人代码中 用 i==0 做出口 是怎么实现的。
我的代码:
#include <cstdio>
#include <cstring>
#include <iostream>
using namespace std; long long nowx;
long long dp[][][];
int num[]; long long dfs(int i, int d, bool okPreZero, bool isQuery) {
//printf("(%d,%d,%s,%s)\n", i, d, okPreZero?"T":"F", isQuery?"T":"F");
long long &nowdp = dp[i][d][okPreZero];
if (!isQuery && ~nowdp) return nowdp;
if (i == ) {
if (d!=) return nowdp = ;
else if (okPreZero) {
return nowdp = ;
} else {
return nowdp = ;
}
}
long long ans = ;
int end = isQuery?num[i-]:;
for (int j = ; j <= end; j++) {
if (okPreZero) {
ans += dfs(i-,j,true,isQuery && j==end);
} else {
if (d == && j == ) {
ans += dfs(i-,j,false,isQuery && j==end);
} else {
ans += dfs(i-, j, true, isQuery && j==end);
}
}
}
long long ten = ;
for (int j = ; j < i-; j++) ten*=;
if (d== && okPreZero) ans += (isQuery?((nowx%ten)+):(ten));
if (!isQuery) nowdp = ans;
return ans;
} long long cal(long long x) {
nowx = x;
if (x == -) return ;
if (x == ) return ;
int len = ;
while (x) {
num[++len] = x%;
x/=;
}
return dfs(len+, , false, true);
} int Nmain() {
long long a;
memset(dp, -, sizeof(dp));
while (cin>>a) {
cout<<"---"<<cal(a)<<endl;
}
return ;
} int main() {
int t;
cin>>t;
int cas = ;
memset(dp, -, sizeof(dp));
while (t--) {
long long m, n;
cin>>m>>n;
cout<<"Case "<<cas++<<": "<<cal(n)-cal(m-)<<endl;
}
return ;
}
不理解的代码:
#include<cstdio>
#include<cstring>
#include<cmath>
typedef long long LL;
LL dp[][][];
int bit[],len;
LL a,b;
LL dfs(int pos,int v,int flag,int limit)
{
if (pos<=) return flag?v:;
if (!limit&&dp[pos][v][flag]!=-) return dp[pos][v][flag];
int end=(limit?bit[pos]:);
LL re=;
for (int i=;i<=end;i++)
{
int tmp;
if (flag&&(i==)) tmp=;else tmp=;
re+=dfs(pos-,v+tmp,flag||i,limit&&(end==i));
}
if (!limit) dp[pos][v][flag]=re;
return re;
} LL solve(LL n)
{ if (n==-) return -;
if (n==) return ;
len=;
while (n)
{ bit[++len]=n%;
n/=;
}
return dfs(len,,,);
}
int main()
{ memset(dp,,sizeof(dp));
int cas,i=;
scanf("%d",&cas);
while (cas--)
{scanf("%lld%lld",&a,&b);
printf("Case %d: %lld\n",++i,solve(b)-solve(a-));
}
return ;
}
LightOJ 1140: How Many Zeroes? (数位DP)的更多相关文章
- LightOJ 1140 How Many Zeroes? (数位DP)
题意:统计在给定区间内0的数量. 析:数位DP,dp[i][j] 表示前 i 位 有 j 个0,注意前导0. 代码如下: #pragma comment(linker, "/STACK:10 ...
- light oj 1140 - How Many Zeroes? 数位DP
思路:dp[i][j]:表示第i位数,j表示是否有0. 代码如下: #include<iostream> #include<stdio.h> #include<algor ...
- UVALive - 6575 Odd and Even Zeroes 数位dp+找规律
题目链接: http://acm.hust.edu.cn/vjudge/problem/48419 Odd and Even Zeroes Time Limit: 3000MS 问题描述 In mat ...
- LightOJ 1032 - Fast Bit Calculations 数位DP
http://www.lightoj.com/volume_showproblem.php?problem=1032 题意:问1~N二进制下连续两个1的个数 思路:数位DP,dp[i][j][k]代表 ...
- lightoj 1021 - Painful Bases(数位dp+状压)
题目链接:http://www.lightoj.com/volume_showproblem.php?problem=1021 题解:简单的数位dp由于总共就只有16个存储一下状态就行了.求各种进制能 ...
- LightOJ 1140 How Many Zeroes
题意:写出一个给定区间的每个数,求出一共写了多少个零. 解法:数位DP,定义dp[len][flag][num]:len的定义为数位的长度,flag定义为前导0和没有前导0的两种状态,num定义为写的 ...
- lightoj 1140 - How Many Zeroes?(数位dp)
Jimmy writes down the decimal representations of all natural numbers between and including m and n, ...
- LightOJ - 1140 统计0的数位 数位DP
注意以下几点: 搜索维度非约束条件的都要记录,否则大概率出错,比如_0 st参数传递和_0的互相影响要分辨清楚 num==-1就要返回0而不是1 #include<iostream> #i ...
- 数位dp(D - How Many Zeroes? LightOJ - 1140 )
题目链接:https://cn.vjudge.net/contest/278036#problem/D 题目大意:T组测试数据,每一次输入两个数,求的是在这个区间里面,有多少个0,比如说19203包括 ...
随机推荐
- Java--equals和 == 的比较和equals()、HashCode()的重写
一. equals和 == 的比较 1.== 运算符 ① == 如果比较的是基本数据类型,则比较的是值. ② == 如果比较的是引用数据类型,则比较的是地址值. 2.equals ①它属于java.l ...
- 【Python学习之五】高级特性1(切片、迭代、列表生成器、生成器、迭代器)
1.切片 有一个list—>L = [1,2,3,4,5,6,7]或tuple—>T = (1,2,3,4,5,6,7),如果想取得前三个元素,怎么操作? 硬方法,也是低效的方法是:L= ...
- 【java】 java 内存解读
具体请参考 vamei java 内存管理和垃圾回收 java的内存分为栈内存和堆内存两部分 栈内存 主要存储一些参数,局部变量和返回地址,参数和局部变量大部分是基本类型的变量,如果是引用类型,实际上 ...
- Python9-loggin模块-day29
什么叫日志日志 是用来记录用户行为或者代码的执行过程 # import logging # logging.debug('debug message') #低级别的 排除信息 # logging.in ...
- PAT Basic 1073
1073 多选题常见计分法 批改多选题是比较麻烦的事情,有很多不同的计分方法.有一种最常见的计分方法是:如果考生选择了部分正确选项,并且没有选择任何错误选项,则得到 50% 分数:如果考生选择了任何一 ...
- Python之多线程与多进程(一)
多线程 多线程是程序在同样的上下文中同时运行多条线程的能力.这些线程共享同一个进程的资源,可以在并发模式(单核处理器)或并行模式(多核处理器)下执行多个任务 多线程有以下几个优点: 持续响应:在单线程 ...
- LOFTER 迁移
title: LOFTER 迁移 date: 2018-09-01 16:41:02 updated: tags: [其他] description: keywords: comments: imag ...
- SVN 删除所有目录下的“.svn”文件夹,让文件夹脱离SVN控制
SVN 删除所有目录下的“.svn”文件夹,将如下语句拷备到记事本,并保存为 *.reg,双击导入注册表,在文件夹右键中就多了一条“Delete SVN Folders”,点击就可以删处此目录下的所有 ...
- 大数据学习——spark学习
计算圆周率 [root@mini1 bin]# ./run-example SparkPi [root@mini1 bin]# ./run-example SparkPi [root@mini1 bi ...
- [uiautomator篇][exist 存在,但click错误]
uiautomator定位页面元素是,定位存在的;但是click的时候,发现点的位置不对,(不知道是android系统的问题还是uiautomator的问题,初步怀疑是系统的问题)