hdu 2089 数位dp
链接:https://vjudge.net/problem/23625/origin
中文,题目不用说了。
其实这题的数据很小,所以直接暴力也可以过,但是还是要学会数位dp,因为并不是每一题的数据都会这么小,这里给出我暴力的代码和数位dp的代码(数位dp其实也是有一个大致的模板的)。
暴力:
#include<stdio.h>
#include<string.h>
int dp[];
int n,m,k,t;
int jug(int a)
{
while(a)
{
if(a%==)
return ;
if(a%==&&a/%==&&a>=)
return ;
a/=;
}
return ;
}
int main()
{
memset(dp,,sizeof(dp));
for(int i=;i<=;i++)
{
if(jug(i))
dp[i]=dp[i-]+;
else
dp[i]=dp[i-];
}
while(scanf("%d%d",&n,&m)!=EOF)
{
if(!n&&!m)
break;
if(m<n)
printf("%d\n",);
else
printf("%d\n",dp[m]-dp[n-]);
}
return ;
}
接下来我只是说一下我的看法,可能不是非常准确,能大致看懂就好。现在假设我们求0到625符合条件的数字有多少个,那么我们把625的每一位存进一个digit数组,先介绍一下我们递归时的顺序(值始终不能超过625),先把百位拿出来,因为百位为6,那么我们可以这样找
先找百位,百位的上界为6,所以百位是受到约束的,最大只能取6,所以百位可以取0,1,2,3,4,5,6。
接下来找十位,0_ _,1_ _,2_ _,3_ _,4_ _,5_ _,前面的这些它们的十位不受约束可以随便取(比如00_,01_,... 09_, 10_,11_...19_ ... 59_),从0到9都可以。
但是到了6(6_ _)就要控制后面的数字了,因为它的十位最大只能取2,所以我们只能取6 0 _,6 1_,6 2 _,同样,在我们递归到了十位时,如果百位是0到5,那么它们的个位也是不受约束的,所以也可以随便取,
但是如果百位是6,那么我们又要看情况了,如果是60_,或者是61_,那么它们的个位也是不受约束的,可以随便取,但是如果是62_,那么它的个位是受到约束的,最大只能到5。
应该差不多了,在程序里,我们从最高位一直递归到个位,同时把当前位不受约束的状态的值保存下来,pos表示现在到了
哪一位(百位?十位?个位?),flag表示到到当前状态为止是否符合条件(0或1),pre表示当前状态的前一位数字是多少(判断是不是有62),limit就是控制上界的关键,我们把它初始化为1,表示目前它的最高位收到限制,在之后的过程中
如果limit为0,那么接下来所有的递归过程里数字都是0到9,都可以取,都可以记忆保存,可能看文字不是特别明白,
还是看看代码吧:
#include<stdio.h>
#include<string.h>
int dp[][][];
int digit[];
int n,m,k,t;
void init()
{
memset(dp,-,sizeof(dp));
}
int dfs(int pos,int flag,int pre,int limit)//当前位,是否满足条件,前一位,控制上界
{
if(!pos)//到了个位,返回值
return flag;
if(!limit&&dp[pos][pre][flag]!=-)//直接输出之前记录的值,节省时间
return dp[pos][pre][flag];
int num=limit?digit[pos]:;//非常关键的东西,如果limit为1,那么这个位置受到约束,最大只能为digit[pos]
int s=; //否则它就不受约束,可以为任何值
for(int i=;i<=num;i++)
{
if(pre==&&i==||!flag||i==)
s+=dfs(pos-,,i,limit&&i==num);//这个limit&&i==num是为了确定下一位数是否受约束
else
s+=dfs(pos-,,i,limit&&i==num);
}
if(!limit&&dp[pos][pre][flag]==-)//记录
dp[pos][pre][flag]=s;
return s; }
int cal(int a)
{
int len=;
memset(digit,,sizeof(digit));
while(a)
{
digit[++len]=a%;
a/=;
}
return dfs(len,,,);
}
int main()
{
init();
while(scanf("%d%d",&n,&m)!=EOF)
{
if(!n&&!m)
break;
printf("%d\n",cal(m)-cal(n-));
}
return n;
}
hdu 2089 数位dp的更多相关文章
- hdu:2089 ( 数位dp入门+模板)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2089 数位dp的模板题,统计一个区间内不含62的数字个数和不含4的数字个数,直接拿数位dp的板子敲就行 ...
- HDU 2089 数位dp入门
开始学习数位dp...一道昨天看过代码思想的题今天打了近两个小时..最后还是看了别人的代码找bug...(丢丢) 传说院赛要取消 ? ... 这么菜不出去丢人也好吧~ #include<stdi ...
- HDU 2089 数位dp/字符串处理 两种方法
不要62 Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submis ...
- 不要62(HDU 2089数位dp入门)
题意:统计区间 [a,b] 中不含 4 和 62 的数字有多少个. 分析:dp[i][f]数字表示不含 4 和 62的前提下,剩余长度为 len ,首位是否为 6 的个数. #include < ...
- HDU - 2089 数位DP 初步
中文题目,不要62和4 从高位往低位DP,注意有界标志limit的传递 dp2记忆有界情况下的计数结果,据说用处不大 我所参考的入门文章就是半搜索(有界)半记忆(无界)的 进阶指南中提出dfs维度有多 ...
- 杭电hdu 2089 数位dp
杭州人称那些傻乎乎粘嗒嗒的人为62(音:laoer). 杭州交通管理局经常会扩充一些的士车牌照,新近出来一个好消息,以后上牌照,不再含有不吉利的数字了,这样一来,就可以消除个别的士司机和乘客的心理障碍 ...
- hdu 2089 数位dp入门题
#include<stdio.h> //dp[i][0]代表不存在不吉利数字 //dp[i][1]代表不存在不吉利数字但是以2开头 //dp[i][2]代表存在不吉利数字 #define ...
- hdu 4507 数位dp(求和,求平方和)
http://acm.hdu.edu.cn/showproblem.php?pid=4507 Problem Description 单身! 依旧单身! 吉哥依旧单身! DS级码农吉哥依旧单身! 所以 ...
- hdu 4352 数位dp + 状态压缩
XHXJ's LIS Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total ...
随机推荐
- kubernetes发布tomcat服务,通过deployment,service布署
1.制作tomcat镜像 参考docker tomcat镜像制作 此处直接拉取 查看已有可镜像 先设置docker阿里源,即添加 "registry-mirrors": [&quo ...
- leetcode998
class Solution: def __init__(self): self.prelist = list() def preTree(self,root:TreeNode): if root ! ...
- 【Source Insight 】之marco学习笔记1
我们学习编程语言都是从Hello World!,现在我们学习marco也不例外 打开C:\Users\%USERPROFILE%\Documents\Source Insight 4.0\Projec ...
- PHP 使用非对称加密算法(RSA)
解释: 非对称加密算法需要两个密钥:公开密钥(publickey)和私有密钥(privatekey).公开密钥与私有密钥是一对,如果用公开密钥对数据进行加密,只有用对应的私有密钥才能解密:如果用私有密 ...
- 配置阿里云的金融云上的rsync
论坛里看到易淘发的教程, 转载过来
- wget 递归下载整个网站(网站扒皮必备)
有时间看到别人网站的页面比较漂亮,就想给扒皮下来,学习学习.分享一个我常用网站扒皮命令wget 这个命令可以以递归的方式下载整站,并可以将下载的页面中的链接转换为本地链接. wget加上参数之后,即可 ...
- linux下安装jdk,tomcat,maven
1. jdk安装 下载jdk的linux版本. >tar -zxvf jdk1.8.0_191.tar.gz 配置环境变量: >vim /etc/profile最前面添加: expor ...
- Django基础介绍
1.web应用 Web应用程序是一种可以通过Web访问的应用程序,程序的最大好处是用户很容易访问应用程序,用户只需要有浏览器即可,不需要再安装其他软件. 应用程序有两种模式C/S.B/S.C/S是客户 ...
- 修改window本地hosts文件,修改域名指向
Hosts是一个没有扩展名的系统文件,可以用记事本等工具打开,其作用就是将一些常用的网址域名与其对应的IP地址建立一个关联“数据库”,当用户在浏览器中输入一个需要登录的网址时,系统会首先自动从Host ...
- react native 中es6语法解析
react native是直接使用es6来编写代码,许多新语法能提高我们的工作效率 解构赋值 var { StyleSheet, Text, View } = React; 这句代码是ES6 中新增的 ...