HDU 3555 Bomb(数位DP模板啊两种形式)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3555
add one point.
Now the counter-terrorist knows the number N. They want to know the final points of the power. Can you help them?
The input terminates by end of file marker.
3 1 50 500
0 1 15 Hint From 1 to 500, the numbers that include the sub-sequence "49" are "49","149","249","349","449","490","491","492","493","494","495","496","497","498","499", so the answer is 15.
题意:
求0 到n的数中有多少个数字是含有‘49’的。
PS:
数位DP
//dp[i][j]:长度为i的数的第j种状态
//dp[i][0]:长度为i可是不包括49的方案数
//dp[i][1]:长度为i且不含49可是以9开头的数字的方案数
//dp[i][2]:长度为i且包括49的方案数
(转)状态转移例如以下
dp[i][0] = dp[i-1][0] * 10 - dp[i-1][1]; // not include 49 假设不含49且,在前面能够填上0-9 可是要减去dp[i-1][1] 由于4会和9构成49
dp[i][1] = dp[i-1][0]; // not include 49 but starts with 9 这个直接在不含49的数上填个9即可了
dp[i][2] = dp[i-1][2] * 10 + dp[i-1][1]; // include 49 已经含有49的数能够填0-9,或者9开头的填4
接着就是从高位開始统计
在统计到某一位的时候,加上 dp[i-1][2] * digit[i] 是显然对的。由于这一位能够填 0 - (digit[i]-1)
若这一位之前挨着49,那么加上 dp[i-1][0] * digit[i] 也是显然对的。
若这一位之前没有挨着49,可是digit[i]比4大,那么当这一位填4的时候,就得加上dp[i-1][1]
代码例如以下:
- #include <cstdio>
- #include <cstring>
- #include <iostream>
- #include <algorithm>
- using namespace std;
- typedef __int64 LL;
- LL dp[27][3];
- int c[27];
- //dp[i][j]:长度为i的数的第j种状态
- //dp[i][0]:长度为i可是不包括49的方案数
- //dp[i][1]:长度为i且不含49可是以9开头的数字的方案数
- //dp[i][2]:长度为i且包括49的方案数
- void init()
- {
- memset(dp,0,sizeof(dp));
- dp[0][0] = 1;
- for(int i = 1; i <= 20; i++)
- {
- dp[i][0] = dp[i-1][0]*10-dp[i-1][1];
- dp[i][1] = dp[i-1][0]*1;
- dp[i][2] = dp[i-1][2]*10+dp[i-1][1];
- }
- }
- int cal(LL n)
- {
- int k = 0;
- memset(c,0,sizeof(c));
- while(n)
- {
- c[++k] = n%10;
- n/=10;
- }
- c[k+1] = 0;
- return k;
- }
- void solve(int len, LL n)
- {
- int flag = 0;//标记是否出现过49
- LL ans = 0;
- for(int i = len; i >= 1; i--)
- {
- ans+=c[i]*dp[i-1][2];
- if(flag)
- {
- ans+=c[i]*dp[i-1][0];
- }
- else if(c[i] > 4)
- {
- //这一位前面没有挨着49。但c[i]比4大,那么当这一位填4的时候,要加上dp[i-1][1]
- ans+=dp[i-1][1];
- }
- if(c[i+1]==4 && c[i]==9)
- {
- flag = 1;
- }
- }
- printf("%I64d\n",ans);
- }
- int main()
- {
- int t;
- LL n;
- init();
- scanf("%d",&t);
- while(t--)
- {
- scanf("%I64d",&n);
- int len = cal(n+1);
- solve(len, n);
- }
- return 0;
- }
DFS版
代码例如以下:
- #include <cstdio>
- #include <cstring>
- #include <iostream>
- #include <algorithm>
- using namespace std;
- #define LL __int64
- LL n, dp[25][3];
- //dp[i][j]:长度为i。状态为j
- int digit[25];
- //nstatus: 0:不含49, 1:不含49但末尾是4, 2 :含49
- LL DFS(int pos, int status, int limit)
- {
- if(pos <= 0) // 假设到了已经枚举了最后一位。而且在枚举的过程中有49序列出现
- return status==2;//注意是 ==
- if(!limit && dp[pos][status]!=-1) //对于有限制的询问我们是不可以记忆化的
- return dp[pos][status];
- LL ans = 0;
- int End = limit?digit[pos]:9; // 确定这一位的上限是多少
- for(int i = 0; i <= End; i++) // 每一位有这么多的选择
- {
- int nstatus = status; // 有点else s = statu 的意思
- if(status==0 && i==4)//高位不含49。而且末尾不是4 ,如今末尾添4返回1状态
- nstatus = 1;
- else if(status==1 && i!=4 && i!=9)//高位不含49。且末尾是4,如今末尾加入的不是4返回0状态
- nstatus = 0;
- else if(status==1 && i==9)//高位不含49,且末尾是4,如今末尾加入9返回2状态
- nstatus = 2;
- ans+=DFS(pos-1, nstatus, limit && i==End);
- }
- if(!limit)
- dp[pos][status]=ans;
- return ans;
- }
- int cal(LL x)
- {
- int cnt = 0;
- while(x)
- {
- digit[++cnt] = x%10;
- x/=10;
- }
- digit[cnt+1] = 0;
- return cnt;
- }
- int main()
- {
- int t;
- scanf("%d",&t);
- while(t--)
- {
- memset(dp,-1,sizeof(dp));
- scanf("%I64d",&n);
- int len = cal(n);
- LL ans = DFS(len, 0, 1);
- printf("%I64d\n",ans);
- }
- return 0;
- }
HDU 3555 Bomb(数位DP模板啊两种形式)的更多相关文章
- HDU 3555 Bomb 数位dp
题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=3555 Bomb Time Limit: 2000/1000 MS (Java/Others) Mem ...
- HDU 3555 Bomb 数位DP 入门
给出n,问所有[0,n]区间内的数中,不含有49的数的个数 数位dp,记忆化搜索 dfs(int pos,bool pre,bool flag,bool e) pos:当前要枚举的位置 pre:当前要 ...
- HDU - 3555 - Bomb(数位DP)
链接: https://vjudge.net/problem/HDU-3555 题意: The counter-terrorists found a time bomb in the dust. Bu ...
- Bomb HDU - 3555 (数位DP)
Bomb HDU - 3555 (数位DP) The counter-terrorists found a time bomb in the dust. But this time the terro ...
- HDU 3555 Bomb (数位DP-记忆化搜索模板)
题意 求区间[1,n]内含有相邻49的数. 思路 比较简单的按位DP思路.这是第一次学习记忆化搜索式的数位DP,确实比递推形式的更好理解呐,而且也更通用~可以一般化: [数位DP模板总结] int d ...
- HDU(3555),数位DP
题目链接:http://acm.split.hdu.edu.cn/showproblem.php?pid=3555 Bomb Time Limit: 2000/1000 MS (Java/Others ...
- hud 3555 Bomb 数位dp
Bomb Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 131072/65536 K (Java/Others) Total Subm ...
- HDU 2089 数位dp/字符串处理 两种方法
不要62 Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submis ...
- hdoj 3555 BOMB(数位dp)
//hdoj 3555 //2013-06-27-16.53 #include <stdio.h> #include <string.h> __int64 dp[21][3], ...
随机推荐
- MySQL常用技能篇
写在之前的话: 之前一直在用MSSERVER,刚用MySQL时有很多的不适应.就此小结一下工作中遇到的问题和场景(用的不是很深入,供初学者参考),文中出现的局限性欢迎指出 MySQL有客户端式(SQL ...
- 【BZOJ4444】国旗计划 - 决策单调性
Description A国正在开展一项伟大的计划——国旗计划.这项计划的内容是边防战士手举国旗环绕边境线奔袭一圈.这项计划需要多名边防战士以接力的形式共同完成,为此,国土安全局已经挑选了N名优秀的边 ...
- C++ vector基本用法
转自金河http://www.cnblogs.com/wang7/archive/2012/04/27/2474138.html 1 基本操作 (1)头文件#include<vector> ...
- Beat 'Em Up Game Starter Kit (横版格斗游戏) cocos2d-x游戏源代码
浓缩精华.专注战斗! 游戏的本质是什么?界面?养成?NoNo! 游戏来源于对实战和比赛的模拟,所以它的本源就是对抗.就是战斗! 是挥洒热血的一种方式! 一个游戏最复杂最难做的是什么?UI?商城? ...
- python json及mysql——读取json文件存sql、数据库日期类型转换、终端操纵mysql及python codecs读取大文件问题
preface: 近期帮师兄处理json文件,须要读到数据库里面,以备其兴许从数据库读取数据.数据是关于yelp站点里面的: https://github.com/Yelp/dataset-examp ...
- mysql-计算字段
一.计算字段 存储在数据库表中的数据一般不是应用程序所需要的格式 1.如果想在一个字段中既显示公司名,又显示公司的地址,但这两个信息一般包含在不同的字段中. 2.城市.州和邮编存储在不同的列中,但邮件 ...
- 去除ListView 上下边界蓝色或黄色阴影
默认的情况下,在 ListView 滑动到顶部或者是底部的时候,会有黄色或者蓝色的阴影出现.在不同的版本号上解决办法是不同的,在 2.3 版本号之前能够在 ListView 的属性中通过设置 andr ...
- 【转】webshell检测——使用auditd进行system调用审计
本文档将介绍:如何通过Linux审计系统auditd监测WebShell执行系统命令的行为. 测试环境:CentOS7.0_x64 auditd简介 Linux审计系统提供了一种跟踪系统上与安全相关的 ...
- hdoj--1301--Jungle Roads(克鲁斯卡尔)
Jungle Roads Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Tot ...
- Mvc前后端显示不同的404错误页
最近做的系统前端是移动端的,后端是PC端,然后404页面不能用通一个,so 查找了一些资料,找到了一个解决办法 在Global.asax文件夹下添加Application_EndRequest事件处 ...