题目链接: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[];
ll dp[][];
int N; ll dfs(int pos,int cnt,bool limit)
{
if(pos==) return cnt; //已经精确到某一个数,那么直接返回这个数有几个“1”.
if(!limit && ~dp[pos][cnt]) return dp[pos][cnt]; //如果曾今计算过dfs(pos,cnt,0),直接返回dp[pos][cnt] int up=limit?dig[pos]:; //确定当前第pos位的枚举上界
int ans=; //定义ans变量,记录累加结果,在函数最后return ans
for(int i=;i<=up;i++) //暴力枚举当前第pos位的数
{
if(i==) ans+=dfs(pos-,cnt+,limit && i==up); //如果当前位为1
else ans+=dfs(pos-,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=;
while(x) //将上界N记录到dig数组中
{
dig[++pos]=x%;
x/=;
}
return dfs(pos,,);
} int main()
{
while(scanf("%d",&N)!=EOF)
{
memset(dp,-,sizeof(dp));
printf("%lld\n",solve(N));
}
}

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

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

    1009 数字1的数量 基准时间限制:1 秒 空间限制:131072 KB   给定一个十进制正整数N,写下从1开始,到N的所有正数,计算出其中出现所有1的个数.   例如:n = 12,包含了5个1 ...

  2. 1009 数字1的数量 数位dp

    1级算法题就这样了,前途渺茫啊... 更新一下博客,我刚刚想套用数位dp的模板,发现用那个模板也是可以做到,而且比第二种方法简单很多 第一种方法:我现在用dp[pos][now]来表示第pos位数字为 ...

  3. 51nod 1042 数字0-9的数量 数位dp

    1042 数字0-9的数量 基准时间限制:1 秒 空间限制:131072 KB 分值: 10 难度:2级算法题  收藏  关注 给出一段区间a-b,统计这个区间内0-9出现的次数.   比如 10-1 ...

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

    题意: 小于等于n的所有数中1的出现次数 分析: 数位DP 预处理dp[i][j]存 从1~以j开头的i位数中有几个1,那么转移方程为: if(j == 1) dp[i][j] = dp[i-1][9 ...

  5. 51nod 1009 数字1的数量

    1009 数字1的数量   给定一个十进制正整数N,写下从1开始,到N的所有正数,计算出其中出现所有1的个数.   例如:n = 12,包含了5个1.1,10,12共包含3个1,11包含2个1,总共5 ...

  6. 51nod 1009 数字1的数量(数位dp模板)

    给定一个十进制正整数N,写下从1开始,到N的所有正数,计算出其中出现所有1的个数. 例如:n = 12,包含了5个1.1,10,12共包含3个1,11包含2个1,总共5个1.   数位dp的模板题   ...

  7. 51Nod 1009:1009 数字1的数量 (思维)

    1009 数字1的数量  基准时间限制:1 秒 空间限制:131072 KB 分值: 5 难度:1级算法题  收藏  关注 给定一个十进制正整数N,写下从1开始,到N的所有正数,计算出其中出现所有1的 ...

  8. 1001 数组中和等于K的数对 1002 数塔取数问题 1003 阶乘后面0的数量 1004 n^n的末位数字 1009 数字1的数量

    1001 数组中和等于K的数对 基准时间限制:1 秒 空间限制:131072 KB 分值: 5 难度:1级算法题 给出一个整数K和一个无序数组A,A的元素为N个互不相同的整数,找出数组A中所有和等于K ...

  9. 51nod 1042数字0-9的数量

    1042 数字0-9的数量  基准时间限制:1 秒 空间限制:131072 KB 分值: 10 难度:2级算法题  收藏  关注 给出一段区间a-b,统计这个区间内0-9出现的次数. 比如 10-19 ...

随机推荐

  1. linux-ubuntu14.04以下使用gdb出现的问题

    问题: (gdb) list 没有符号表被读取. 请使用 "file" 命令. 原因事实上说的比較清楚,可运行文件里没有符号表,为什么会没有符号表呢.由于符号表是在编译过程中使用的 ...

  2. mongodb安装使用笔记

    mongodb安装使用 安装后配置环境变量 创建数据库文件夹并连接数据库,并执行mongod --dbpath c:\workname 打开新的cmd,执行mongo命令,管理数据库 show dbs ...

  3. 给自己的android扫盲文 - 1

    1. 你得知道,android开发打一开始就是java的事,没其它语言什么事情,就是说google提供的android sdk中的api都是java的api2. 至于强大的跨平台语言,你懂的,非c/c ...

  4. backbone学习笔记:视图(View)

    Backbone 视图对象主要用来渲染数据,监听事件. Backbone的视图对象可以展示Model数据,也可以把用户编辑的Model数据传递到后台,可以通过监听事件操作视图里的DOM元素 举例: v ...

  5. 通过RF数据库查询中文字段结果正常显示的转换方法

    方法1:统一显示为中文 1.通过RF数据库查询中文字段结果格式:'\xba\xcb\xbc\xf5\xcd\xa8\xb9\xfd' 2.通过Decode Bytes To String进行gbk解码 ...

  6. Windows下POSIX线程编程(pThread)环境搭建

    系统: Windows 编辑器:codeblocks13.12 1. 简介: Windows有一个叫 POSIX Threads for Win32 的开源项目给出了一个功能比较完善的Windows下 ...

  7. 二叉查找树(BST)的性质

    二叉查找树的性质: 1.左子树上所有结点的值均小于或等于它的根结点的值. 2.右子树上所有结点的值均大于或等于它的根结点的值. 3.左.右子树也分别为二叉排序树. 下图中这棵树,就是一颗典型的二叉查找 ...

  8. c++ map使用问题【运行结果不一样】

    map经常把指针作为key,这种情况下. 我们经常会很自然的以为,如果要取元素时,会按照我们存的顺序拿到元素. 但是事实上不是这样的,因为map取得时候是按key的大小排序的,而如果用指针作为key, ...

  9. Android 全屏Activity以透明的对话框形式弹出

    1. styles.xml <style name="transcutestyle" parent="@android:style/Theme.DeviceDefa ...

  10. Oracle中V$SESSION等各表的字段解释,Oracle官方解释

    一.常用的视图 1.会话相关视图 View Description V$PROCESS Contains information about the currently active processe ...