51nod 1009 - 数字1的数量 - [数位DP][模板的应用以及解释]

题目链接:https://www.51nod.com/onlineJudge/questionCode.html#!problemId=1009

基准时间限制:1 秒 空间限制:131072 KB

给定一个十进制正整数N,写下从1开始,到N的所有正数,计算出其中出现所有1的个数。

例如:n = 12,包含了5个1。1,10,12共包含3个1,11包含2个1,总共5个1。

Input

输入N(1 <= N <= 10^9)

Output

输出包含1的个数

Input示例

12

Output示例

5

题解:

这道题跟前面的HDU2089HDU3555这两道题有一点不同,dp数组的设定不是很一样。

本题解的前提是,默认为阅读过“数位dp总结 之 从入门到模板 http://blog.csdn.net/wust_zzwh/article/details/52100392”;

首先我们可以清楚地定义好,输入一个上界,那么它的位数就是“最大位数”,暂且定义为len;

然后我们的pos,定义域是在[0,len],它其实不是准确指定某一个长度的数;

而是说,它现在这个dp[pos][…]值,我们只管到范围从第1位到第pos位,再宽的话,抱歉我管不了。

首先,我们说清楚“位”:

  

接下来我们要设计dfs函数;

dfs(pos,cnt,limit)代表:

  已知第pos+1位到第len位上已有cnt个“1”(这是状态state);

  已知我现在要进行枚举的第pos位有没有上界限制(有上界限制的话上界up = dig[pos],没有的话上界up = 9);

  然后这个函数就能给你返回:第1位到第pos位上枚举所有可能出现的数字(每一位都可任意填0~9),连接上前面第pos+1位到第len位上已经确定下来的数字,共出现了多少个“1”。

例如:

  N = 1200,len = 4,pos= 2,limit = 0,已确定第3~4位是“01”(即cnt=1,出现了一个“1”),

  第1位到第2位上就可以从00枚举到99,那么dfs(2,1,0)的返回值就是0100到0199出现了多少个“1”;

这样设计的一个原因是,我们必须要求我们的dfs(pos,cnt,limit)能够在传入参数后,能迭代返回我们需要的答案。

换句话说,样例输入N = 1200,那么配合一个dig数组(dig[1]=0,dig[2]=0,dig[3]=2,dig[4]=1),我们的dfs(4,0,1)函数就能给你正确答案。

接下来是关于dp数组的定义:

dp数组可以说是dfs函数的记忆化,可以说dp[pos][cnt]的值就等于dfs(pos,cnt,0);

只要你算过一次dfs(pos,cnt,0),就用dp[pos][cnt]给你记录下来,在没有上界限制的情况下,可以直接使用dp[pos][cnt],而不用再去dfs(pos,cnt,0),节省大量时间。

例如:

  继续用前面的例子,N = 1200,len = 4,pos= 2,limit = 0,

  确定第3~4位是“01”(即cnt=1,出现了一个“1”),或者第3~4位是“10”(依然cnt=1,出现了一个“1”),

  第1位到第2位上从00枚举到99,那么“0100到0199”和“1000到1099”出现的“1”的个数是一样的,

  那么我们记录下“0100到0199”的第一次dfs(2,1,0)的返回值,存储在dp[2][1],那么之后要去算“1000到1099”的时候直接返回dp[2][1]即可;

然后设计dfs()函数的转移:

  dfs(pos,cnt,limit) 等于:暴力枚举第pos位上的数:i = 0 ~ ( limit ? dig[pos] : 9 ),累加起所有dfs( pos - 1, cnt + ( i==1?1:0) , limit && i == dig[pos] ).

最后!……配上数位DP的板子,就很好写啦。

AC代码:

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
int dig[11];
ll dp[11][11];
int N;

ll dfs(int pos,int cnt,bool limit)
{
    if(pos==0) return cnt; //已经精确到某一个数,那么直接返回这个数有几个“1”.
    if(!limit && ~dp[pos][cnt]) return dp[pos][cnt]; //如果曾今计算过dfs(pos,cnt,0),直接返回dp[pos][cnt]

    int up=limit?dig[pos]:9; //确定当前第pos位的枚举上界
    int ans=0; //定义ans变量,记录累加结果,在函数最后return ans
    for(int i=0;i<=up;i++) //暴力枚举当前第pos位的数
    {
        if(i==1) ans+=dfs(pos-1,cnt+1,limit && i==up); //如果当前位为1
        else ans+=dfs(pos-1,cnt,limit && i==up); //如果当前位不为1
    }

    if(!limit) dp[pos][cnt]=ans; //将dfs(pos,cnt,0)记录到dp[pos][cnt]
    return ans;
}
ll solve(int x)
{
    int pos=0;
    while(x) //将上界N记录到dig数组中
    {
        dig[++pos]=x%10;
        x/=10;
    }
    return dfs(pos,0,1);
}

int main()
{
    while(scanf("%d",&N)!=EOF)
    {
        memset(dp,-1,sizeof(dp));
        printf("%lld\n",solve(N));
    }
}

原文地址:https://www.cnblogs.com/dilthey/p/8503376.html

时间: 2024-10-09 05:50:18

51nod 1009 - 数字1的数量 - [数位DP][模板的应用以及解释]的相关文章

51nod 1009 数字1的数量 数位dp

1009 数字1的数量 基准时间限制:1 秒 空间限制:131072 KB 给定一个十进制正整数N,写下从1开始,到N的所有正数,计算出其中出现所有1的个数. 例如:n = 12,包含了5个1.1,10,12共包含3个1,11包含2个1,总共5个1. Input 输入N(1 <= N <= 10^9) Output 输出包含1的个数 Input示例 12 Output示例 5 #include<bits/stdc++.h> using namespace std; #define

51Nod 1009 数字1的个数 | 数位DP

题意: 小于等于n的所有数中1的出现次数 分析: 数位DP 预处理dp[i][j]存 从1~以j开头的i位数中有几个1,那么转移方程为: if(j == 1) dp[i][j] = dp[i-1][9]*2+pow(10,i-1);else dp[i][j] = dp[i-1][9]+dp[i][j-1]; 然后注意下对于每个询问统计的时候如果当前位为1需要额外加上他后面所有位数的个数,就是n%pow(10,i-1); 这样总复杂度log(n)*10 #include <bits/stdc++.

51nod 1042 数字0-9的数量 (数位dp、dfs、前导0)

1042 数字0-9的数量 基准时间限制:1 秒 空间限制:131072 KB 分值: 10 难度:2级算法题 收藏 关注 取消关注 给出一段区间a-b,统计这个区间内0-9出现的次数. 比如 10-19,1出现11次(10,11,12,13,14,15,16,17,18,19,其中11包括2个1),其余数字各出现1次. Input 两个数a,b(1 <= a <= b <= 10^18) Output 输出共10行,分别是0-9出现的次数 Input示例 10 19 Output示例

51Nod 1042 数字0-9的数量(数位DP)

题意: 求[l,r]中数字0-9分别出现的次数,11算两次1 思路: 数位dp题解好难写,直接贴代码吧 dp[i]表示[0, 10^i-1]中出现j的次数(按i位补全前导0,显然0-9出现的次数是相同的) 最后再减去每一位出现的前导零即可 代码: #include<iostream> #include<cstdio> #include<algorithm> #include<cmath> #include<cstring> #include<

51Nod 1009 数字1的数量

具体题解发一下大佬的分析http://blog.csdn.net/wyg1997/article/details/52169036 1 #include <iostream> 2 #include <algorithm> 3 using namespace std; 4 typedef long long LL; 5 6 int main(){ 7 std::ios::sync_with_stdio(false); 8 int n; 9 cin >> n; 10 LL

[51 nod]1009 数字1的数量

1009 数字1的数量 基准时间限制:1 秒 空间限制:131072 KB 分值: 5 难度:1级算法题 给定一个十进制正整数N,写下从1开始,到N的所有正数,计算出其中出现所有1的个数. 例如:n = 12,包含了5个1.1,10,12共包含3个1,11包含2个1,总共5个1. Input 输入N(1 <= N <= 10^9) Output 输出包含1的个数 Input示例 12 Output示例 5详解请看大牛博客:http://www.cnblogs.com/jy02414216/ar

POJ 3286 How many 0&#39;s(数位DP模板)

题目链接:http://poj.org/problem?id=3286 题目大意: 输入n,m,求[n,m]的所有数字中,0出现的总数是多少,前导零不算. 解题思路: 模板题,设dp[pos][num],pos为数位,num为当前0的数目,然后套数位DP模板即可. 还有之前的一些思考: 关于数位DP求0时,dp下标记录num有什么作用,num不是与后面的0的个数无关吗?是的,在(!limit&&!lead)的情况下,前面有多少0是不影响后面可以出现多少0的.但是,比如说dp[pos][nu

CodeVS 1359 数字计数 51nod 1042 数字0-9的数量 Pascal

题目大意: 我的代码又臭又长,但是毕竟是我这个jr想了几天才推出的公式,看别的大神都写数位DP,所以我决定分享一下我的思路.我认为我的思路一向是最好理解的! 要分两种情况讨论: 1.0的情况.我们首先推出0~9中只有1个0,在0~99中有(1+(9*100*1))个0{第一位可以为1~9,第二位可以为0~9,0只可以放在后者,所以乘1},在0~999中有((1+(9*100*1))+(9*101*2))个0~6666为例,先算出0~999中0的个数,在算出1000~5999中0的个数,则为(10

数位dp模板 [dp][数位dp]

现在才想到要学数位dp,我是不是很弱 答案是肯定的 以一道自己瞎掰的题为模板 1 //题: 2 //输入数字n 3 //从0枚举到n,计算这n+1个数中含有两位数a的数的个数 4 //如12930含有两位数93 5 #include<cstdio> 6 #include<cstring> 7 #include<iostream> 8 using namespace std; 9 10 int n,m=0,a,g1,g2; 11 int dp[10][10][2],lim