数位DP入门题——[hdu2089]不要62
数位DP是我的噩梦。
现在初三了,却没AC过数位DP的题目。
感觉数位DP都是毒瘤……
题目
hdu不用登录也可以进去,所以就不把题目copy到这里来了。
题目大意
求区间[n,m][n,m][n,m]中,不含有444和626262的数的个数。
解析
数位DP的难点主要在于不能出界。
因为这个东西,我被卡了不知道多少年……
先不要想出界,那么显然,状态可以这么设:
设fi,jf_{i,j}fi,j表示做到第iii位,并且这一位为jjj的方案数。
方程显然
if (j!=4 && !(j==6 && k==2))
f[i][j]+=f[i-1][k];
所以我们可以先预处理出fff数组。
对于区间[n,m][n,m][n,m],类似前缀和,我们可以将其转化成求[1,x][1,x][1,x]。
接下来就是最毒瘤的出界问题了。
比如x=314x=314x=314
那么f2,0,f2,1,f2,2f_{2,0},f_{2,1},f_{2,2}f2,0,f2,1,f2,2可以计入贡献,因为前面的数字不管怎么变,都必定不会出界。
但是f2,3f_{2,3}f2,3是不可以直接计入贡献的。
所以咋办?
我们可以继续下一位,计算x=14x=14x=14时的贡献,当然,要记录一下上一次最后的数,在计入贡献前要先判断是否合法。
因为我们计算的都是小于xxx的贡献,所以在计算之前我们要先将xxx加一
具体见下面的代码。
代码
using namespace std;
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
#define MAXM 1000000
int n,m;
int f[10][10];
int pow10[10];
void init();
int getans(int);
int main(){
pow10[0]=1;
for (int i=1;i<=9;++i)
pow10[i]=pow10[i-1]*10;
init();
do{
scanf("%d%d",&n,&m);
if (n==0 && m==0)
return 0;
printf("%d\n",getans(m)-getans(n-1));
}
while (1);
return 0;
}
void init(){
for (int i=0;i<=9;++i)
f[0][i]=(i!=4);//i!=4时就为1
for (int i=1;i<=6;++i)
for (int j=0;j<=9;++j){
if (j==4)
continue;
for (int k=0;k<=9;++k)
f[i][j]+=f[i-1][k];
if (j==6)
f[i][j]-=f[i-1][2];//将j==6和k==2不合法的贡献减掉
}
}
int getans(int x){
x++;//因为下面的算法计算的是小于x的贡献,所以要先x++
if (x<0)
return 0;
int res=0,w=log10(x);//w为x的位数
for (int i=w,lst=0;i>=0;--i){
int s=x/pow10[i]%10;//求出第i为的贡献
for (int j=0;j<s;++j)
res+=f[i][j];//将小于s的贡献全部加上(因为前面的数怎么变都不可能越界)
if (lst==6 && 2<s)
res-=f[i][2];//同样将不合法的贡献减去
if (s==4 || lst==6 && s==2)//如果这个时候出现不合法的情况,那么后面的都是不合法的,直接退出
break;
lst=s;
}
return res;
}
–
总结
数位DP最难的地方就是判断出界。
所以我们应该从高位到低位逐个判断。
保证范围小于边界。
数位DP入门题——[hdu2089]不要62的更多相关文章
- HDU 2089 不要62【数位DP入门题】
不要62 Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submis ...
- HDU 2089 - 不要62 - [数位DP][入门题]
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2089 Time Limit: 1000/1000 MS (Java/Others) Memory Li ...
- 【数位dp】【HDU 3555】【HDU 2089】数位DP入门题
[HDU 3555]原题直通车: 代码: // 31MS 900K 909 B G++ #include<iostream> #include<cstdio> #includ ...
- 数位DP入门题
站点一览: hdu 2089"不要62" hdu 4734"F(X)" poj 3252"Round Numbers" hdu 3709&q ...
- 数位DP入门(A - 不要62 HDU - 2089 &&B - Bomb HDU - 3555 )
题目链接:https://cn.vjudge.net/contest/278036#problem/A 具体思路:对于给定的数,我们按照位数进行运算,枚举每一位上可能的数,在枚举的时候需要注意几个条件 ...
- hdu 2089 数位dp入门题
#include<stdio.h> //dp[i][0]代表不存在不吉利数字 //dp[i][1]代表不存在不吉利数字但是以2开头 //dp[i][2]代表存在不吉利数字 #define ...
- hdu3555 Bomb (数位dp入门题)
Bomb Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 131072/65536 K (Java/Others)Total Submi ...
- xbz分组题B 吉利数字 数位dp入门
B吉利数字时限:1s [题目描述]算卦大湿biboyouyun最近得出一个神奇的结论,如果一个数字,它的各个数位相加能够被10整除,则称它为吉利数.现在叫你计算某个区间内有多少个吉利数字. [输入]第 ...
- 数位dp入门 hdu2089 不要62
数位dp入门 hdu2089 不要62 题意: 给定一个区间[n,m] (0< n ≤ m<1000000),找出不含4和'62'的数的个数 (ps:开始以为直接暴力可以..貌似可以,但是 ...
随机推荐
- MySQL 其他基础知识
-- 查询存储引擎show engines;-- 显示可用存储引擎show variables like 'have%'; -- concat多个字段联合select tname ,cname ,co ...
- 线段树+欧拉函数——cf1114F
调了半天,写线段树老是写炸 /* 两个操作 1.区间乘法 2.区间乘积询问欧拉函数 欧拉函数计算公式 phi(mul(ai))=mul(ai) * (p1-1)/p1 * (p2-1)/p2 * .. ...
- 如何区分无线AP跟无线路由器
无线AP是一个无线网络的接入点,俗称“热点”.主要有路由交换接入一体设备和纯接入点设备,一体设备执行接入和路由工作,纯接入设备只负责无线客户端的接入,纯接入设备通常作为无线网络扩展使用,与其他AP或者 ...
- SpringCloud学习笔记(六):Feign+Ribbon负载均衡
简介 官网解释: http://projects.spring.io/spring-cloud/spring-cloud.html#spring-cloud-feign Feign是一个声明式WebS ...
- Redis Cluste部署
一.原生搭建篇Cluster了解cluster的架构 Redis-cluster是使用的是一致性哈希算法来切分数据存储,总计16383个槽,分成16383/N(redis节点)个分区,存取时将key转 ...
- TIB、TEB 信息
https://en.wikipedia.org/wiki/Win32_Thread_Information_Block 这是重点 Position Length Windows Versions D ...
- [转]Java多线程
一.线程的生命周期及五种基本状态 关于Java中线程的生命周期,首先看一下下面这张较为经典的图: 上图中基本上囊括了Java中多线程各重要知识点.掌握了上图中的各知识点,Java中的多线程也就基本上掌 ...
- python2 和Python3 转unicode占位。
- 19-11-06-&
你&我处于这里……在一起? $$\text{%%%Wearry}$$ ZJ: 一遇到Wearry的思维题就得×得够呛. 考试心态炸裂,码上三个暴力然后就不知道该干啥了. 现在就想敲自己. 不要 ...
- 19-10-29-Night-X
布谷. 欢迎大家来不耻下问. 这里是正解不会,暴力写跪,乱搞鸡肋的某虻 . 好想放假啊!!!! 话说猫厂现在产量低迷…… ZJ一下: T1,三维偏序,只码了$\Theta(N^2)$暴力 T2,暴力愉 ...