数位dp从会打模板到不会打模板
打了几个数位$dp$,发现自己除了会打模板之外没有任何长进,遇到非模板题依然什么都不会
那么接下来这篇文章将介绍如何打模板(滑稽)
假设我们要处理$l----r$
采用记忆化搜索的方式,枚举$<=r$每一种情况,枚举每一位,然后再枚举$<=l-1$每一种情况,然后两个值相减即可
我们可以比较轻松打出一个模板
ll dfs(ll x,ll pre,ll lead,ll limit){
if(x>pos) return 1;
if(!limit&&f[x][pre]) return f[x][pre];
ll ans=0;
ll mx=limit?maxn[pos-x+1]:9;
for(ll i=0;i<=mx;i++){
if(????????) continue;
if(lead&&i==0) ans+=dfs(x+1,-2,1,limit&&(i==mx));
else ans+=dfs(x+1,i,0,limit&&(i==mx));
}
if(!limit&&!lead) f[x][pre]=ans;
return ans;
}
解释一下$limit$是什么
假设有一个数
$1\ 2\ 3\ 4\ 5\ 6$
$1\ 2\ 3\ ?\ ?\ ?$
我们枚举到第四位时最多枚举到$4$,
$1\ 2\ 3\ 4\ 5\ 6$
$1\ 2\ 2\ ?\ ?\ ?$
这时我们枚举到第四位最多枚举到$9$
$limit$就是判断这个的
那么为什么要在$!limit$下才记忆化呢?
如果在所有情况下我们都记录f,那么假如之前枚举到$9$时你记录了一个答案,然后当前位有$limit$限制根本枚举不到$9$,你仍然用了这个f会出现错误
当然你这样记忆化也可以这样避免冲突
ll dfs(ll x,ll limit,ll tmp,ll d){
if(f[x][limit][tmp][d])
return f[x][limit][tmp][d];
………………
f[x][limit][tmp][d]=ans;
return ans;
}
解释一下$lead$
处理前导0用的,
有的时候
$0\ 0\ 0\ 4\ 5\ 6$
也被认为是合法的,这时处理可能会出现问题,$lead$特判特殊处理
那么我们看几道模板题
例题
windy数
Windy 定义了一种 Windy 数:不含前导零且相邻两个数字之差至少为2的正整数被称为 Windy 数。
Windy 想知道,在l和r 之间,包括l 和 r ,总共有多少个 Windy 数?
非常简单对不对
套模板
代码:
#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define A 50
ll f[A][A],maxn[A];
ll pos=0,a,b;
ll dfs(ll x,ll pre,ll lead,ll limit){
if(x>pos) return 1;
if(!limit&&f[x][pre]) return f[x][pre];
ll ans=0;
ll mx=limit?maxn[pos-x+1]:9;
// printf("mx=%lld\n",mx);
for(ll i=0;i<=mx;i++){
if(abs(pre-i)<2) continue;
if(lead&&i==0) ans+=dfs(x+1,-2,1,limit&&(i==mx));
else ans+=dfs(x+1,i,0,limit&&(i==mx));
// printf("ans=%lld\n",ans);
}
if(!limit&&!lead) f[x][pre]=ans;
return ans;
}
ll solve(ll x){
pos=0;
memset(f,0,sizeof(f));
while(x){
maxn[++pos]=x%10;
x/=10;
}
ll ans=dfs(1,-2,1,1);
return ans;
}
int main(){
scanf("%lld%lld",&a,&b);
swap(a,b);
printf("%lld\n",solve(a)-solve(b-1));
}
不要62
$l---r$间数位上不含$4$且没有$62$相连的数个数
套模板
代码
#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define A 50
ll f[A][A],maxn[A];
ll pos=0,a,b;
ll dfs(ll x,ll pre,ll limit){
if(!x) return 1;
if(!limit&&f[x][pre]) return f[x][pre];
// printf("x=%lld pre=%lld limit=%lld\n",x,pre,limit);
ll ans=0;
ll mx=limit?maxn[x]:9;
for(ll i=0;i<=mx;i++){
if(pre==6&&i==2) continue;
if(i==4) continue;
ans+=dfs(x-1,i,limit&&(i==mx));
}
// printf("ans=%lld\n",ans);
if(!limit) f[x][pre]=ans;
return ans;
}
ll solve(ll x){
pos=0;
memset(f,0,sizeof(f));
while(x){
maxn[++pos]=x%10;
x/=10;
}
ll ans=dfs(pos,0,1);
return ans;
}
int main(){
a=233,b=233;
while(a!=0&&b!=0){
scanf("%lld%lld",&a,&b);
if(a<b)swap(a,b);
if(a==0||b==0){
return 0;
}
printf("%lld\n",solve(a)-solve(b-1));
}
}
手机号码
至少$3$个相邻的相同的数,$8$ $4$不能同时出现
套模板,注意下特判,不然会70到自闭
代码
#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define A 30
ll f[A][A][A][2][2][2],pos[A];
ll tot=0,a,b;
ll dfs(ll x,ll prer,ll pre,ll limit,ll _4,ll _8,ll ok){
if(_4&&_8) return 0;
if(x>tot&&!ok) return 0;
if(x>tot&&ok) return 1;
if(!limit&&f[x][prer][pre][_4][_8][ok]) return f[x][prer][pre][_4][_8][ok];
ll maxn=limit?pos[tot-x+1]:9;
ll ans=0;
for(ll i=0;i<=maxn;i++){
if(x==1&&i==0) continue;
if(_4&&i==8) continue;
if(_8&&i==4) continue;
if(pre==prer&&i==pre&&i==prer)
ans+=dfs(x+1,pre,i,(limit&&i==maxn),(_4||i==4),(_8||i==8),1);
else
ans+=dfs(x+1,pre,i,limit&&(i==maxn),(_4||i==4),(_8||i==8),ok);
// printf("ans=%lld i=%lld maxn=%lld\n",ans,i,maxn);
}
if(!limit)
f[x][prer][pre][_4][_8][ok]=ans;
return ans;
}
ll solve(ll x){
tot=0;
if(x<1e10) return 0;
memset(f,0,sizeof(f));
while(x){
pos[++tot]=x%10;
x/=10;
}
ll ans=dfs(1,0,0,1,0,0,0);
return ans;
}
int main(){
scanf("%lld%lld",&a,&b);
if(a<b) swap(a,b);
printf("%lld\n",solve(a)-solve(b-1));
}
花神的数论题
设 sum(i) 表示 i 的二进制表示中 1 的个数。给出一个正整数 N ,花神要问你派(Sum(i)),也就是 sum(1)—sum(N) 的乘积。
套模板,等等,我要套什么,
这个题不算很板,值得思考思考。
数位dp计算出所有二进制下sum个数,然后快速幂处理一下,
代码
#include<bits/stdc++.h>
using namespace std;
#define A 53
#define mod 10000007
#define ll long long
ll n,tot=0,sum=0;
ll f[A][2][A][A],ans[A],pos[A];
ll dfs(ll cur,ll up,ll tmp,ll d){
if(!cur)
return tmp==d;
if(~f[cur][up][tmp][d])
return f[cur][up][tmp][d];
ll lim=up?pos[cur]:1;
ll ret=0;
for(ll i=0;i<=lim;i++)
ret+=dfs(cur-1,up&&i==lim,tmp+(i==1),d);
return f[cur][up][tmp][d]=ret;
}
ll meng(ll a,ll b){
ll ret=1;
while(b)
ret=ret*(b&1?a:1)%mod,a=a*a%mod,b>>=1;
return ret;
}
ll solve(ll x){
sum=1;
while(x){
pos[++tot]=x&1;
x>>=1;
}
for(ll i=1;i<=tot;i++){
memset(f,-1,sizeof(f));
ans[i]=dfs(tot,1,0,i);
}
for(ll i=1;i<=tot;i++){
sum=(sum*max(meng(i,ans[i]),1ll))%mod;
}
return sum;
}
int main(){
scanf("%lld",&n);
printf("%lld\n",solve(n));
}
剩下一些题都不算很板
题解慢慢补充八
那么你现在会模板了吗?
数位dp从会打模板到不会打模板的更多相关文章
- 数位dp模板 [dp][数位dp]
现在才想到要学数位dp,我是不是很弱 答案是肯定的 以一道自己瞎掰的题为模板 //题: //输入数字n //从0枚举到n,计算这n+1个数中含有两位数a的数的个数 //如12930含有两位数93 #i ...
- [模板] 数位dp
数位dp 简介 数位dp指满足特定性质的数的计数, 如求 \([l, r]\) 区间内不含 \(2\) 的数的个数. 一般来说, 数位dp利用dfs解决, 有时状态数较多, 需要hash表优化. 模板 ...
- 51nod 1009 数字1的数量(数位dp模板)
给定一个十进制正整数N,写下从1开始,到N的所有正数,计算出其中出现所有1的个数. 例如:n = 12,包含了5个1.1,10,12共包含3个1,11包含2个1,总共5个1. 数位dp的模板题 ...
- 数位dp讲解及模板
转载自:传送门 数位DP其实是很灵活的,所以一定不要奢求一篇文章就会遍所有数位DP的题,这一篇只能是讲清楚一种情况,其他情况遇到再总结,在不断总结中慢慢体会这个思想,以后说不定就能达到一看到题目就能灵 ...
- 51nod 1009 - 数字1的数量 - [数位DP][模板的应用以及解释]
题目链接:https://www.51nod.com/onlineJudge/questionCode.html#!problemId=1009 基准时间限制:1 秒 空间限制:131072 KB 给 ...
- HDU 2089 不要62(数位dp模板题)
http://acm.hdu.edu.cn/showproblem.php?pid=2089 题意:求区间内不包含4和连续62的数的个数. 思路: 简单的数位dp模板题.给大家推荐一个好的讲解博客.h ...
- POJ 3286 How many 0's(数位DP模板)
题目链接:http://poj.org/problem?id=3286 题目大意: 输入n,m,求[n,m]的所有数字中,0出现的总数是多少,前导零不算. 解题思路: 模板题,设dp[pos][num ...
- 数位dp小结以及模板
这里是网址 别人的高一啊QAQ.... 嗯一般记忆化搜索是比递推好写的所以我写的都是dfs嗯......(因为我找不到规律啊摔,还是太菜.....) 显然这个东西的条件是非常的有套路..但是不管怎么样 ...
- 数位dp总结 之 从入门到模板
转发自WUST_WenHao巨巨的博客 基础篇 数位dp是一种计数用的dp,一般就是要统计一个区间[le,ri]内满足一些条件数的个数.所谓数位dp,字面意思就是在数位上进行dp咯.数位还算是比较好听 ...
- 数位dp(模板+例题)
文章参考:数位dp之总结 首先,什么是数位dp?它是干什么的? 数位dp是一种计数用的dp,一般就是要统计一个区间[le,ri]内满足一些条件数的个数. 举个栗子: 加入我们要枚举所有上界不超过231 ...
随机推荐
- Jenkins + Docker + ASP.NET Core自动化部署
本来没想着要写这篇博客,但是在实操过程中,一个是被网络问题搞炸了心态(真心感觉网络能把人搞疯,别人下个包.下个镜像几秒钟搞定,我看着我的几KB小水管真是有苦说不出),另一个就是这里面坑还是有一些的,写 ...
- unapp一键登录
一.整理思路 un-app官网提供多种实现[一键登录](https://uniapp.dcloud.net.cn/uniCloud/univerify "")的方法,这里的选择是 ...
- 逆向工程初步160个crackme-------2
有了第一个crackme的经验后,这个crackme用了半个小时就验证成功了.(思路和第一个crackme相似) 动态调试工具:ollydbg (2.10) 文件分析工具:PEID (0.95) 同样 ...
- Pytorch_Part3_模型模块
VisualPytorch beta发布了! 功能概述:通过可视化拖拽网络层方式搭建模型,可选择不同数据集.损失函数.优化器生成可运行pytorch代码 扩展功能:1. 模型搭建支持模块的嵌套:2. ...
- [刷题] 200 Number of Islands
要求 给定一个二维数组,只有0和1两个字符,1代表陆地,0代表水域,纵向和横向的陆地连接成岛屿,被水域隔开,求出地图中有多少岛屿 思路 对要查找的区域进行双重循环遍历,寻找陆地 从陆地初始点开始进行深 ...
- [OS] 概述&学习资料
计算机启动 启动自检 初始化启动 启动加载 内核装载 登录 中断 硬件中断 I/O设备 CPU Timer:时间片结束后,发中断给CPU Scheduler:将CPU合理分配任务使用 异常中断 内存: ...
- bash调试 脚本第一行 加set -x #!/bin/expact -d
#!/bin/bash set -x 脚本第一行加 #!/bin/expact -d
- linux操作系统优化系列-RAID不同阵列模式的选择
背景 笔者所在的某通信运营商某大数据项目由于应用面临瓶颈需要扩充服务器设备,当初上这个项目的时候,服务器上线前的工作(配置raid,安装操作系统,Infiniband网络调试,系统漏洞安全加固)都是我 ...
- Ubuntu 15.04下安装Docker
最近听说Docker很火,不知道什么东西,只知道是一个容器,可以跨平台.闲来无事,我也来倒弄倒弄.本文主要介绍:Ubuntu下的安装,以及基本的入门命令介绍:我的机器是Ubuntu 15.04 64位 ...
- linux中级之HAProxy基础配置
一.haproxy简介 HAProxy是一款提供高可用性.负载均衡以及基于TCP(第四层)和HTTP(第七层)应用的代理软件,HAProxy是完全免费的.借助HAProxy可以快速并且可靠的提供基于T ...