递推版的数位dp

绝对的暴力美学

我们设\(dp[l][i][j][0/1][0/1][0/1]\)表示到了第\(l\)位,这一位上选择的数是\(i\),\(l-1\)位选择的数是\(j\),第一个\(0/1\)代表\(4\)没有/有出现过,第二个\(0/1\)代表\(8\)没有/有出现过,第三个\(0/1\)代表连续三位没有/有出现过

于是转移很简单了

但是卡位实在是鬼畜

我卡位的方式有些鬼畜,所以细节非常的多

#include<iostream>
#include<cstring>
#include<cstdio>
#define re register
#define maxn 16
#define LL long long
LL L,R;
int a[maxn],num;
LL dp[maxn][11][11][2][2][2];
//位数,这一位上的数,上一位的数,0/1表示4/8有/没有出现,0/1表示有/没有连续三位
inline LL slove(LL x)
{
memset(dp,0,sizeof(dp));
memset(a,0,sizeof(a));
num=0;
while(x)
{
a[++num]=x%10;
x/=10;
}//分解数位
a[num+1]=-11,a[num+2]=11;
for(re int i=0;i<=9;i++)
for(re int j=0;j<=9;j++)
for(re int k=0;k<=9;k++)
{
int opt_4=0,opt_8=0;
if(i==4||j==4||k==4) opt_4=1;
if(i==8||j==8||k==8) opt_8=1;
if(i==j&&j==k) dp[3][i][j][opt_4][opt_8][1]+=1;
else dp[3][i][j][opt_4][opt_8][0]+=1;
}//先初始化dp[3]之后再往下推
for(re int l=3;l<num;l++)//刷表转移
for(re int i=0;i<=9;i++)
for(re int j=0;j<=9;j++)
for(re int k=0;k<=9;k++)
for(re int o4=0;o4<=1;o4++)
for(re int o8=0;o8<=1;o8++)
for(re int o=0;o<=1;o++)
dp[l+1][i][j][(i==4)||o4][(i==8)||o8][((i==j)&&(j==k))||o]+=dp[l][j][k][o4][o8][o];
//方程其实挺简单的,就是看看这一位有没有4/8/连续三位就好了
LL ans=0;
for(re int i=3;i<num;i++)//从位数小于给定数的开始
for(re int j=1;j<=9;j++)
for(re int k=0;k<=9;k++)
ans+=dp[i][j][k][0][0][1]+dp[i][j][k][1][0][1]+dp[i][j][k][0][1][1];
for(re int i=1;i<a[num];i++)//位数和给定数相等,但是首位比较小
for(re int j=0;j<=9;j++)
ans+=dp[num][i][j][0][0][1]+dp[num][i][j][1][0][1]+dp[num][i][j][0][1][1];
int o4=0,o8=0,o=0;// 4/8/连续三位有没有出现过
if(a[num]==4) o4=1;
if(a[num]==8) o8=1;
for(re int l=num-1;l>=3;l--)//卡位,这里保证从[l+1,num]和给定数是完全相等的
{
if(o4&&o8) break;
for(re int i=0;i<a[l];i++)
{
int flag=o;
if(i==a[l+1]&&a[l+1]==a[l+2]) o=1;//由于我们选择这一位有可能会导致和上面的两位重复,于是这里需要判断一下,如果有那么就o=1接下来就算没有选出连续三位也可以了
for(re int j=0;j<=9;j++)
{
int cnt=o;
if(i==j&&i==a[l+1]) o=1;
//和上面两位重合的情况
if(o)
{
if(o4&&!o8) ans+=dp[l][i][j][1][0][1]+dp[l][i][j][0][0][1]
+dp[l][i][j][0][0][0]+dp[l][i][j][1][0][0];
if(!o4)
{
if(!o8) ans+=dp[l][i][j][0][0][1]+dp[l][i][j][0][0][0]
+dp[l][i][j][1][0][1]+dp[l][i][j][1][0][0]
+dp[l][i][j][0][1][1]+dp[l][i][j][0][1][0];
if(o8) ans+=dp[l][i][j][0][1][1]+dp[l][i][j][0][1][0]
+dp[l][i][j][0][0][1]+dp[l][i][j][0][0][0];
}
}
if(!o)
{
if(o4&&!o8) ans+=dp[l][i][j][1][0][1]+dp[l][i][j][0][0][1];
if(!o4)
{
if(!o8) ans+=dp[l][i][j][0][0][1]+dp[l][i][j][1][0][1]+dp[l][i][j][0][1][1];
if(o8) ans+=dp[l][i][j][0][1][1]+dp[l][i][j][0][0][1];
}
}
o=cnt;
}
o=flag;//将o还原回来
}
o4=o4||(a[l]==4);
o8=o8||(a[l]==8);
o=o||((a[l]==a[l+1])&&(a[l+1]==a[l+2]));
//判断有没有连续三位4/8出现过
}
if(o4&&o8) return ans;
for(re int i=0;i<=9;i++)
for(re int j=0;j<=9;j++)//卡最后两位
{
if(i*10+j>a[2]*10+a[1]) continue;//不能超过给定数的最后两位
if(o4&&o8) continue;
if(i==4||j==4)
if(o8) continue;//有4就不能有8
if(i==8||j==8)
if(o4) continue;//有8就不能有4
if(i==4&&j==8) continue;
if(i==8&&j==4) continue;//更不可能同时出现
if(o||(i==j&&j==a[3])||(i==a[3]&&a[3]==a[4])) ans++;
//最后两位仍有可能和前面构成三位连续
}
return ans;
}
int main()
{
scanf("%lld%lld",&L,&R);
printf("%lld\n",slove(R)-slove(L-1));
return 0;
}

【[CQOI2016]手机号码】的更多相关文章

  1. 4521: [Cqoi2016]手机号码

    4521: [Cqoi2016]手机号码 Time Limit: 10 Sec Memory Limit: 512 MB Submit: 1030 Solved: 609 [Submit][Statu ...

  2. [BZOJ4521][CQOI2016]手机号码(数位DP)

    4521: [Cqoi2016]手机号码 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 875  Solved: 507[Submit][Status ...

  3. [Bzoj4521][Cqoi2016]手机号码(数位dp)

    4521: [Cqoi2016]手机号码 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 870  Solved: 505[Submit][Status ...

  4. [CQOI2016]手机号码 数位DP

    [CQOI2016]手机号码 用来数位DP入门,数位DP把当前是否需要限制取数范围(是否正在贴着临界值跑,即下面的limited)和一切需要满足的条件全部塞进记忆化搜索参数里面就好了,具体情况转移便好 ...

  5. P4124 [CQOI2016]手机号码

    P4124 [CQOI2016]手机号码 题解 数位DP   DFS  虽然套路,但还是恶心到找不到锅在哪里 注意这个 然后你就发现其实这样就不用记录前导0了 锅在这个鬼地方QAQ 代码 #inclu ...

  6. [Luogu P4124] [CQOI2016]手机号码 (数位DP)

    题面 传送门:洛咕 Solution 感谢神仙@lizbaka的教学 这题是数位DP的非常非常模板的题目,只是状态有点多 . 这题我使用记忆化搜索实现的 中国有句古话说的好,有多少个要求就设多少个状态 ...

  7. BZOJ4521: [Cqoi2016]手机号码

    Description 人们选择手机号码时都希望号码好记.吉利.比如号码中含有几位相邻的相同数字.不含谐音不 吉利的数字等.手机运营商在发行新号码时也会考虑这些因素,从号段中选取含有某些特征的号 码单 ...

  8. BZOJ4521 Cqoi2016 手机号码 【数位DP】

    Description 人们选择手机号码时都希望号码好记.吉利.比如号码中含有几位相邻的相同数字.不含谐音不吉利的数字等.手机运营商在发行新号码时也会考虑这些因素,从号段中选取含有某些特征的号码单独出 ...

  9. 【洛谷P4124】[CQOI2016]手机号码

    手机号码 数位DP模板题 记忆化搜索: #include<iostream> #include<cstring> #include<cstdio> using na ...

  10. [BZOJ4521][Cqoi2016]手机号码 (数位dp)

    题目描述 人们选择手机号码时都希望号码好记.吉利.比如号码中含有几位相邻的相同数字.不含谐音不吉利的数字等.手机运营商在发行新号码时也会考虑这些因素,从号段中选取含有某些特征的号码单独出售.为了便于前 ...

随机推荐

  1. DataGridView 绑定数据方法

    DataGridView控件用于显示来自多种外部数据源中的数据,用户可以在此控件添加行和列,并可以填充数据.   如要让DataGridView显示数据库中的数据,只需要将此控件绑定到挑用数据库的数据 ...

  2. MySQL (一)(未完成)

    并发控制 读写锁 读锁: 共享锁 写锁: 排它锁 颗粒度 表锁,MySQL中开销最小的锁 行锁,MySQL中开销最大的锁 事务 ACID特性 原子性(Automatic) 隔离性(Isolation) ...

  3. Spark内部结构详解

    参考: https://github.com/JerryLead/SparkInternals/blob/master/markdown/english/5-Architecture.md?winzo ...

  4. 轻松学习java可重入锁(ReentrantLock)的实现原理(转 图解)

    前言 相信学过java的人都知道 synchronized 这个关键词,也知道它用于控制多线程对并发资源的安全访问,兴许,你还用过Lock相关的功能,但你可能从来没有想过java中的锁底层的机制是怎么 ...

  5. grunt-contrib-watch 监控 JS 文件改变来运行预定义的Tasks

    依赖于 GruntJs ~0.4.0 监控 JS 文件改变来运行预定义的Tasks Demo: watch: { scripts: { files: ['src/**/*.js'], tasks: [ ...

  6. POJ P2104 K-th Number

    You are working for Macrohard company in data structures department. After failing your previous tas ...

  7. hdu5824 graph

    传送门 题意:定义一个无向图的权值为图中形为树的连通块数量的$k$次方,求所有$n$个点有标号的简单无向图的权值之和. 这个题还是很妙的啊……(好吧,其实只有最后的复合函数求导比较有意思……) 先套路 ...

  8. websocket协议及案例

    WebSocket是一种用于在服务器与客户端之间实现高效的双向通信的机制.可以解决数据实时性要求比较高的应用,比如:在线聊天,在线教育,炒股或定位等. 一:websocket产生背景: 为了解决这种实 ...

  9. 【一些简单的jQuery选择器】

    学习[js DOM 编程艺术],最后面有许多jQuery的选择器,每个都动手敲了一遍. jQuery 提供了高级选择器的方法. js获取元素的三个基本方法分别是通过标签名,类名和id,即(getEle ...

  10. C++学习笔记(9)----关于变量和数组大小的一道容易出错的面试题

    一道容易出错的C++笔试题 求下面代码的输出内容: int main(int argc,char* argv[]) { char str1[]="Hello"; char* str ...