当前数位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)的更多相关文章

  1. LightOJ 1140 How Many Zeroes? (数位DP)

    题意:统计在给定区间内0的数量. 析:数位DP,dp[i][j] 表示前 i 位 有 j 个0,注意前导0. 代码如下: #pragma comment(linker, "/STACK:10 ...

  2. light oj 1140 - How Many Zeroes? 数位DP

    思路:dp[i][j]:表示第i位数,j表示是否有0. 代码如下: #include<iostream> #include<stdio.h> #include<algor ...

  3. UVALive - 6575 Odd and Even Zeroes 数位dp+找规律

    题目链接: http://acm.hust.edu.cn/vjudge/problem/48419 Odd and Even Zeroes Time Limit: 3000MS 问题描述 In mat ...

  4. LightOJ 1032 - Fast Bit Calculations 数位DP

    http://www.lightoj.com/volume_showproblem.php?problem=1032 题意:问1~N二进制下连续两个1的个数 思路:数位DP,dp[i][j][k]代表 ...

  5. lightoj 1021 - Painful Bases(数位dp+状压)

    题目链接:http://www.lightoj.com/volume_showproblem.php?problem=1021 题解:简单的数位dp由于总共就只有16个存储一下状态就行了.求各种进制能 ...

  6. LightOJ 1140 How Many Zeroes

    题意:写出一个给定区间的每个数,求出一共写了多少个零. 解法:数位DP,定义dp[len][flag][num]:len的定义为数位的长度,flag定义为前导0和没有前导0的两种状态,num定义为写的 ...

  7. lightoj 1140 - How Many Zeroes?(数位dp)

    Jimmy writes down the decimal representations of all natural numbers between and including m and n, ...

  8. LightOJ - 1140 统计0的数位 数位DP

    注意以下几点: 搜索维度非约束条件的都要记录,否则大概率出错,比如_0 st参数传递和_0的互相影响要分辨清楚 num==-1就要返回0而不是1 #include<iostream> #i ...

  9. 数位dp(D - How Many Zeroes? LightOJ - 1140 )

    题目链接:https://cn.vjudge.net/contest/278036#problem/D 题目大意:T组测试数据,每一次输入两个数,求的是在这个区间里面,有多少个0,比如说19203包括 ...

随机推荐

  1. NOIP模拟赛 经营与开发 小奇挖矿

    [题目描述] 4X概念体系,是指在PC战略游戏中一种相当普及和成熟的系统概念,得名自4个同样以“EX”为开头的英语单词. eXplore(探索) eXpand(拓张与发展) eXploit(经营与开发 ...

  2. java中的jdbc操作

    package demo; import java.sql.Connection; import java.sql.DriverManager; import java.sql.PreparedSta ...

  3. 如何下载并安装 robomongo 到Ubuntu 系统

    官网下载软件,https://robomongo.org/download wget https://download.robomongo.org/1.2.1/linux/robo3t-1.2.1-l ...

  4. Ubuntu 18.04安装显卡驱动

    安装完双系统,我遇到好几次开机或关机有问题,导致我重装了3次系统,第三次我才知道是显卡驱动问题,Ubuntu预装的开源Nvidia驱动太烂了,需要换官方驱动. 把 nouveau 驱动加入黑名单 $s ...

  5. CentOS 7 忘记root密码解决方法

    CentOS 7  root密码的重置方式和CentOS 6完全不一样,CentOS 7与之前的版本6变化还是比较大的,以进入单用户模式修改root密码为例: 1.重启机器,进入grub菜单的时候按e ...

  6. 多线程辅助类之CountDownLatch(三)

    CountDownLatch信号灯是一个同步辅助类,在完成一组正在其他线程中执行的操作之前,它允许一个或多个线程一直等待.它可以实现多线程的同步互斥功能,和wait和notify方法实现功能类似,具体 ...

  7. debian软raid

    http://www.linuxidc.com/Linux/2013-06/86487.htm  

  8. 《鸟哥的Linux私房菜》学习笔记(2)——Bash特性

    一.shell的基本概念:                               shell 意思是外壳,它是离用户最近的程序.shell提供用户操作系统的接口,我们通过shell将输入的命令与 ...

  9. luogu3370 【模板】字符串哈希

    #include <algorithm> #include <iostream> #include <cstring> #include <cstdio> ...

  10. jmeter进行dubbo接口测试

    最近工作中接到一个需求,需要对一个MQ消息队列进行性能测试,测试其消费能力,开发提供了一个dubbo服务来供我调用发送消息. 这篇博客,介绍下如何利用jmeter来测试dubbo接口,并进行性能测试. ...