题目链接:

http://acm.hdu.edu.cn/showproblem.php?pid=3555

Bomb

Time Limit: 2000/1000 MS (Java/Others)
Memory Limit: 131072/65536 K (Java/Others)
#### 问题描述
> The counter-terrorists found a time bomb in the dust. But this time the terrorists improve on the time bomb. The number sequence of the time bomb counts from 1 to N. If the current number sequence includes the sub-sequence "49", the power of the blast would 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 first line of input consists of an integer T (1
> The input terminates by end of file marker.
#### 输出
> For each test case, output an integer indicating the final points of the power.
#### 样例
> **sample input**
> 3
> 1
> 50
> 500
>
> **sample output**
> 0
> 1
> 15

题意

给你一个数n,求1到n里面有多少个数其中存在子串49的。

题解

数位dp。

代码

#include<iostream>
#include<cstdio>
#include<cstring>
#include<vector>
using namespace std; const int maxn = 20;
typedef long long LL; //dp[len][0]表示长度<=len,且不包含49的总数
//dp[len][1]表示长度<=len,且以9结尾但不包含49的总数
//dp[len][2]表示长度<=len,且包好49的总数
//dp[len][0]+dp[len][2]=所有长度小于等于len的数。
//dp[len][1]是dp[len][0]的一个子集。
LL dp[maxn][3];
void pre() {
memset(dp, 0, sizeof(dp));
dp[0][0] = 1;
for (int i = 1; i < maxn; i++) {
dp[i][0] = dp[i - 1][0] * 10 - dp[i - 1][1];
dp[i][1] = dp[i - 1][0];
dp[i][2] = dp[i - 1][2] * 10 + dp[i - 1][1];
}
} int main() {
pre();
int tc;
scanf("%d", &tc);
while (tc--) {
LL x; scanf("%I64d", &x);
LL ans = 0;
int arr[maxn], tot=0;
while (x) { arr[++tot]=x%10; x /= 10; }
bool flag = false; //标记x的高位时否已经出现49的组合。
for (int i = tot; i >= 1; i--) {
//这时考虑的是高位已经固定,这一位数为0(是可以有0的,因为高位还会有数,虽然最高位后面每数了,但它统计的是<=len,
//而不是==len!!!)到arr[i]-1的所有情况中包好49的数
ans += dp[i-1][2] * arr[i];
if (flag) ans += dp[i-1][0] * arr[i];
else if (arr[i] > 4) {//如果4刚好是arr[i],那也是不能乱搞的!边界情况我们会一直往后推来考虑
ans += dp[i - 1][1];
}
if (i + 1 <= tot&&arr[i] == 9 && arr[i + 1] == 4) {
flag = 1;
}
}
if (flag) ans++;
printf("%I64d\n", ans);
}
return 0;
}

再来一发dfs:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<vector>
using namespace std; const int maxn = 20;
typedef __int64 LL; //dp[len][0]是给第len+1位不为4的时候用的
//dp[len][1]是给第len+1位为4的时候用的(有点像私人订制)
//我们只保存没有限制的状态,对于有限制的状态每次都要算,不过没有限制的状态出现的要不有限制的多的多。
LL dp[maxn][2],ten[maxn];
LL n; int arr[maxn];
LL dfs(int len, int is4, int ismax) {
if (len == 0) return 0;
if (!ismax&&dp[len][is4] >= 0) return dp[len][is4];
LL res = 0;
int ed = ismax ? arr[len] : 9;
for (int i = 0; i <= ed; i++) {
if (i == 9 && is4) {
//这里是可以直接算的哦
res += ismax ? (n%ten[len - 1] + 1) : ten[len - 1];
}
else {
res += dfs(len - 1, i == 4, ismax&&i == ed);
}
}
return ismax ? res : dp[len][is4] = res;
} LL solve(LL x) {
int tot = 0;
while (x) { arr[++tot] = x % 10; x /= 10; }
return dfs(tot, false, true);
} void init() {
memset(dp, -1, sizeof(dp));
ten[0] = 1;
for (int i = 1; i < maxn; i++) ten[i] = ten[i - 1] * 10;
} int main() {
int tc;
scanf("%d", &tc);
init();
while (tc--) {
scanf("%I64d", &n);
printf("%I64d\n", solve(n));
}
return 0;
}

杂七杂八

orzorzorz

HDU 3555 Bomb 数位dp的更多相关文章

  1. HDU 3555 Bomb 数位DP 入门

    给出n,问所有[0,n]区间内的数中,不含有49的数的个数 数位dp,记忆化搜索 dfs(int pos,bool pre,bool flag,bool e) pos:当前要枚举的位置 pre:当前要 ...

  2. HDU - 3555 - Bomb(数位DP)

    链接: https://vjudge.net/problem/HDU-3555 题意: The counter-terrorists found a time bomb in the dust. Bu ...

  3. Bomb HDU - 3555 (数位DP)

    Bomb HDU - 3555 (数位DP) The counter-terrorists found a time bomb in the dust. But this time the terro ...

  4. HDU(3555),数位DP

    题目链接:http://acm.split.hdu.edu.cn/showproblem.php?pid=3555 Bomb Time Limit: 2000/1000 MS (Java/Others ...

  5. HDU 3555 Bomb (数位DP-记忆化搜索模板)

    题意 求区间[1,n]内含有相邻49的数. 思路 比较简单的按位DP思路.这是第一次学习记忆化搜索式的数位DP,确实比递推形式的更好理解呐,而且也更通用~可以一般化: [数位DP模板总结] int d ...

  6. hud 3555 Bomb 数位dp

    Bomb Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 131072/65536 K (Java/Others) Total Subm ...

  7. hdoj 3555 BOMB(数位dp)

    //hdoj 3555 //2013-06-27-16.53 #include <stdio.h> #include <string.h> __int64 dp[21][3], ...

  8. 数位DP入门之hdu 3555 Bomb

    hdu 3555 Bomb 题意: 在1~N(1<=N<=2^63-1)范围内找出含有 ‘49’的数的个数: 与hdu 2089 不要62的区别:2089是找不不含 '4'和 '62'的区 ...

  9. HDU 3555 Bomb(数位DP模板啊两种形式)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3555 Problem Description The counter-terrorists found ...

随机推荐

  1. Vue.js学习 Item12 – 内部响应式原理探究

    深入响应式原理 大部分的基础内容我们已经讲到了,现在讲点底层内容.Vue.js 最显著的一个功能是响应系统 —— 模型只是普通对象,修改它则更新视图.这让状态管理非常简单且直观,不过理解它的原理也很重 ...

  2. nginx+php与apache+php性能对比

    测试工具http_load相同的动态页面测试,相同的硬件资源,相同并发,相同请求数量的前提下,nginx+php比apache+php的性能要 差,而且如果请求的压力大于硬件资源的承受能力,nginx ...

  3. ASP.NET的错误处理机制之一(概念)

    对Web应用程序来说,发生不可预知的错误和异常在所难免,我们必须为Web程序提供错误处理机制.当错误发生时,我们必须做好两件事情:一是将错误信息记录日志,发邮件通知网站维护人员,方便技术人员对错误进行 ...

  4. mysql中存不进去json_encode格式的数据

    主要是因为json_encode格式的数据,中间带有\,在存入数据库的时候,会把反斜杠删除了. 所以,想要存进去的话,需要在外层调用一下函数addslashes();这个函数会在每个反斜杠的前面添加反 ...

  5. BufferedReader,缓冲输入字符流

    1. /** * 输入字符流: * --------|Reader 所有输入字符流的基类.抽象类 * -----------|FileReader 读取文件字符串的输入字符流. * --------- ...

  6. How to display SSRS report based on customer/Vendor specific language [AX2012]

    Common requirement is to show the reports in customer’s language. [example : Quotations, sales confi ...

  7. Python核心编程--学习笔记--5--数字

    本章的主题是Python中的数字,这里详细介绍每一种数字类型,它们适用的各种运算符,以及用于处理数字的内建函数.在本章的末尾简单介绍了几个标准库中用于处理数字的模块. 1 数字类型 数字:标量贮存,可 ...

  8. java基本概念

    什么是环境变量? 环境变量通常是指在操作系统当中,用来指定操作系统运行时需要的一些参数.通常为一系列的键值对. path环境变量的作用 path环境变量是操作系统外部命令搜索路径 什么是外部命令搜索路 ...

  9. C 构造一个 简单配置文件读取库

    前言 最近看到这篇文章, json引擎性能对比报告 http://www.oschina.net/news/61942/cpp-json-compare?utm_source=tuicool 感觉技术 ...

  10. Python 文件I/O

    文件I/O是Python中最重要的技术之一,在Python中对文件进行I/O操作是非常简单的. 1.打开文件 语法: open(name[, mode[, buffering]]) 1.1文件模式 1 ...