数位dp真·浅谈 By cellur925
预警:由于是从$Vergil$学长那里和$Mathison$大神那里学来的,所以清一色记忆化搜索!qwq
巨佬的数位dp讲解(未来的咕咕日报头条):
https://www.luogu.org/blog/virus2017/shuweidp
数位dp嘛,顾名思义...就是与每个数上的那位有关系...(废话)
一般问题形式都是,在区间$[l,r]$中,满足xx条件的数有多少。然后数据范围一般巨大,枚举是不可能的,这辈子是不可能的。所以一般我们都把它当做字符串处理,预处理出每一位的数(数位),即之后的$border$数组。先求出1~r内满足范围的数,在求出1~l-1范围内的数,运用前缀和的思想一减的。
关于处理成字符串:
一种高精处理方法
scanf("%s",p+);
int lenp=strlen(p+);
int last=lenp;
while(p[last]==''&&last) last--;
for(int i=last+;i<=lenp;i++) p[i]='';
p[last]--;
for(int i=;i<=lenp;i++) a[lenp-i+]=p[i]-'';
memset(dp,-,sizeof(dp));
ans1=dfs(lenp,,);
scanf("%s",q+);
int lenq=strlen(q+);
for(int i=;i<=lenq;i++) a[lenq-i+]=q[i]-'';
memset(dp,-,sizeof(dp));
ans2=dfs(lenq,,);
printf("%d\n",ans2-ans1);
另一种暴力搞,longlong范围内才可。
l--;
while(l)
{
border[++len]=l%;
l/=;
}
memset(dp,-,sizeof(dp));
ans1=dfs(len,,);
len=; while(r)
{
border[++len]=r%;
r/=;
}
memset(dp,-,sizeof(dp));
ans2=dfs(len,,);
len=;
printf("%lld\n",ans2-ans1);
记忆化搜索?其实感觉他是dp的本质吧...(说错了不要打我啊qwq)。搜索我们要遍历所有状态,遍历完就溜,啥也不记下来,没心没肺;dp我们也要遍历全部状态,但是长心眼了,懂得存起来,用空间换时间;然后dp是和记忆化搜索学的,所以记忆化搜索也记录了搜过的状态,当现在搜的之前搜过,就直接调用之前的结果。Update:Vergil学长表示他认为dp的本质是最短路 把各状态当作图的节点,然后与开始节点连的边为初始值...
既然我们记搜了,那么肯定我们需要知道每次记录哪些量:一般地,我们会记到$pos$(现在在第几位,个人习惯从右往左为第1位到第$len$位)、$pre$(上一位是哪个数字,用于与上一位有关的情况)、$ling$前导零标记、$limit$最高位标记、$st$是否已经满足要求了,还有其他实际情况需要的。开始从主程序进来的时候,$ling$和$limit$都是1。
图自@Mathison真·大神
另外,当$limit=1$不能记录和调用dp值!当$ling=1$也不能调用记录dp值!
以上是概论,现在分两种题型扯几道例题。
第一种题型:问你数位满足是否有xx数 or 没有xx数
- 例1 hdu2089不要62
题目大意:问你$[l,r]$中不含4且不含连续62的数的个数。
真·入门题。因为不含,所以只要有4或62连续,之后我们就不用搜了。
#include<cstdio>
#include<algorithm>
#include<cstring> using namespace std;
typedef long long ll; int l,r,len;
int border[];
ll ans1,ans2,dp[][]; ll dfs(int pos,int pre,int limit)
{
if(pos<) return ;
if(!limit && dp[pos][pre]!=-) return dp[pos][pre];
ll tmp=;
ll lim=limit ? border[pos] : ;
for(int i=;i<=lim;i++)
{
if(i==) continue;
if(i==&&pre==) continue;
tmp+=dfs(pos-,i,(limit&&i==lim));
}
if(!limit) dp[pos][pre]=tmp;
return tmp;
} int main()
{
while(scanf("%d%d",&l,&r)!=EOF&&l!=&&r!=)
{
l--;
while(l)
{
border[++len]=l%;
l/=;
}
memset(dp,-,sizeof(dp));
ans1=dfs(len,,);
len=; while(r)
{
border[++len]=r%;
r/=;
}
memset(dp,-,sizeof(dp));
ans2=dfs(len,,);
len=;
printf("%lld\n",ans2-ans1);
}
return ;
}
- 例2 hdu3555Bomb
题目大意:问你1~n中含连续49的数的个数。
数位dp入门题。因为题目问的是“含”,所以我们必须记一个$st$,看之前是否满足。这里就不能像上一道题一样不满足直接$continue$掉了。因为之后还有希望满足啊qwq。
//求1~n 含有连续49的数字有多少
#include<cstdio>
#include<algorithm>
#include<cstring> using namespace std;
typedef long long ll; int T,len;
int border[];
ll dp[][][];
char sta[]; ll dfs(int pos,int pre,int st,int limit)
{
if(pos<) return st;
if(!limit && dp[pos][pre][st]!=-) return dp[pos][pre][st];
ll tmp=;
ll lim=limit ? border[pos] : ;
for(int i=;i<=lim;i++)
tmp+=dfs(pos-,i,st||(i==&&pre==),limit&&i==lim);
if(!limit) dp[pos][pre][st]=tmp;
return tmp;
} int main()
{
scanf("%d",&T);
while(T--)
{
scanf("%s",sta+);
len=strlen(sta+);
for(int i=;i<=len;i++)
border[i]=sta[len-i+]-'';
memset(dp,-,sizeof(dp));
printf("%lld\n",dfs(len,,,));
}
return ;
}
- 例3 poj3208 Apocalypse Someday
第2种题型:问你数位是否满足一些比较复杂的要求(就不是上面那么简单了)
- 例1 [SCOI2009]Windy数
- 例2[ZJOI2010]数字计数
- 例3 LuoguP3413萌数
- 例4 hihocoder#1791幸运数字
- 例5 bzoj1799[AHOI2009]同类分布
- 例6 某qbxt测试题
例题会尽量都补上的qwq
关于输出方案:本来想试着输出下方案的,后来发现实际输出的时候其实会少很多。因为我们记忆化!当一个状态下的值已经确定了,我们就不再切身实际地搜,而是直接调用之前结果了,也就方案会少输出了。当然,没记忆化的时候,显然我们可以输出全部方案。
数位dp。。。按Mathison大佬的话,还是非常套路模板化的,唯一要注意的是举一反三和前导零的处理...。
数位dp真·浅谈 By cellur925的更多相关文章
- 【hdu3652】数位dp(浅尝ACM-A)
向大佬学习 第一次写博客有点紧张,也算是小萌新的突破吧 这次主要是总结一下校内的ACM比赛的各种题,主要是新思路以及学到的新知识 先放一张 下面开始说正事 题面 A wqb-number, or B- ...
- 真·浅谈treap树
treap树是一种平衡树,它有平衡树的性质,满足堆的性质,是二叉搜索树,但是我们需要维护他 为什么满足堆的性质?因为每个节点还有一个随机权值,按照随机权值维持这个堆(树),可以用O(logn)的复杂度 ...
- 线性DP LIS浅谈
LIS问题 什么是LIS? 百度百科 最长上升子序列(Longest Increasing Subsequence,LIS),在计算机科学上是指一个序列中最长的单调递增的子序列. 怎么求LIS? O( ...
- Ural1057. Amount of Degrees 题解 数位DP
题目链接: (请自行百度进Ural然后查看题号为1057的那道题目囧~) 题目大意: Create a code to determine the amount of integers, lying ...
- 浅谈数位DP
在了解数位dp之前,先来看一个问题: 例1.求a~b中不包含49的数的个数. 0 < a.b < 2*10^9 注意到n的数据范围非常大,暴力求解是不可能的,考虑dp,如果直接记录下数字, ...
- 数位dp浅谈(hdu3555)
数位dp简介: 数位dp常用于求区间内某些特殊(常关于数字各个数位上的值)数字(比如要求数字含62,49): 常用解法: 数位dp常用记忆化搜索或递推来实现: 由于记忆化搜索比较好写再加上博主比较蒟, ...
- 浅谈状态压缩DP
浅谈状态压缩DP 本篇随笔简单讲解一下信息学奥林匹克竞赛中的状态压缩动态规划相关知识点.在算法竞赛中,状压\(DP\)是非常常见的动规类型.不仅如此,不仅是状压\(DP\),状压还是很多其他题目的处理 ...
- LuoguP2602 [ZJOI2010]数字计数【数位dp】By cellur925
题目传送门 题目大意:给定两个正整数a和b,求在[a,b]中的所有整数中,每个数码(digit)各出现了多少次. 继续数位dp=w=. 这一次我们不需要记录$pre$啦!(撒花). 因为这次我们需要的 ...
- LuoguP2657 [SCOI2009]windy数 【数位dp】By cellur925
题目传送门 题目大意:在A和B之间,包括A和B,总共有多少个不含前导零且相邻两个数字之差至少为2的正整数? 显然是数位dp啦=w=. 显然与上一位有关,我们$dfs$的时候就要记录$pre$.因为这是 ...
随机推荐
- 读书笔记-2java虚拟机的可达性算法与finalize方法
JAVA通过可达性分析算法来确定堆中哪些对象是应该被回收的. 非常多人包含我曾经在不了解的时候总以为是通过引用计数器来推断某个对象是否应该被回收可是后来想了想包含查阅一些资料发现不是这种.由于假设採用 ...
- 通视频URL截取第一帧图片
为了方便直接给UIImage加个类别,以后什么时候使用可以直接调用. #import <UIKit/UIKit.h> @interface UIImage (Video) /** 通过视频 ...
- TestNG – Dependency Test
转自:http://www.mkyong.com/unittest/testng-tutorial-7-dependency-test/ In TestNG, we use dependOnMetho ...
- Golang 现有的哲学中,要求你尽量手工处理所有的错误返回
更优雅的 Golang 错误处理 - Go语言中文网 - Golang中文社区 https://studygolang.com/articles/9407
- SAP系统更新模块
SAP 系统中,一些单据保存到数据库用的是 update mudule function. 命名是ME_UPDATE_* (业务说明) 例:PR save module: ME_UPDATE_REQU ...
- (linux)wake_lock机制
Android的休眠唤醒主要基于wake_lock机制,只要系统中存在任一有效的wake_lock,系统就不能进入深度休眠,但可以进行设备的浅度休眠操作.wake_lock一般在关闭lcd.tp但 ...
- 基于Vuejs的搜索匹配功能
最近一直在看vue,查了很多资料,看了很多文档和博客,大概半知半解了,然后利用所理解的知识写了一个简单的搜索匹配功能. 大概长这个样子: <!DOCTYPE html> <htm ...
- INFO: Ignoring response <403 https://movie.douban.com/top250>: HTTP status code is not handled or not allowed
爬取豆瓣电影top250,出现以下报错: 2018-08-11 22:02:16 [scrapy.core.engine] INFO: Spider opened 2018-08-11 22:02:1 ...
- Oracle基础:表空间名称大小写问题
现场环境: 操作系统:windows Oracle版本:10g 今天在通过imp导入数据时,日志提示TS_W5_D表空间不存在.感觉很奇怪,导入用户的表空间是ts_w5_d,并 ...
- Linux下使用pip安装keras
Keras是一个底层使用Theano或TensorFlow的深度学习框架,它的设计参考了Torch,用Python语言编写,也很方便使用Python调用,是一个高度模块化的神经网络库,支持使用GPU和 ...