P2188 小Z的 k 紧凑数 题解(数位DP)
2024-09-05 22:37:49
题目链接
解题思路
数位DP,把每一个数位的每一个数对应的可能性表示出来,然后求\(num(1,r)-num(1,l-1)\),其中\(num(i,j)\)表示\([i,j]\)区间里符合要求的数的个数。
其中,\(dp[i][j]\)表示第\(i\)位数字为\(j\)的选择种数。
计算的时候,比如\(num(456)\),就拆开为\(num(1,99)+num(100,399)+num(400,449)+num(450,455)+num(456,456)\)
AC代码
#include<stdio.h>
long long k,dp[20][14],l,r;
int absf(int a){
if(a<0)return -a;
return a;
}
void dpf(){
int i,j,m;
for(i=0;i<=9;i++)dp[0][i]=1;//个位数,初始化为1
for(i=1;i<20;i++)//这是总共的位数
for(j=0;j<=9;j++)//这是这一位
for(m=0;m<=9;m++)//这是上一位
if(absf(j-m)<=k)dp[i][j]+=dp[i-1][m];//这一位和上一位满足条件则加上
}
long long num(long long x){
int n[20]={0},cnt=0,i,j;
long long ans=0;
while(x>0){
n[cnt++]=x%10;
x/=10;
}
//首位为0
for(i=0;i<cnt-1;i++)
for(j=1;j<=9;j++)
ans+=dp[i][j];
//首位为[1,n[cnt-1])
if(cnt>0)for(i=1;i<n[cnt-1];i++)ans+=dp[cnt-1][i];
//首位为n[cnt-1]
for(i=cnt-2;i>=0;i--){
for(j=0;j<n[i];j++){
if(absf(n[i+1]-j)<=k)ans+=dp[i][j];
}
if(absf(n[i+1]-n[i])>k)break;
//非常重要!!前几位已经不满足绝对值之差不大于k之后就不能再继续下去了
if(!i&&absf(n[i+1]-j)<=k)ans+=dp[i][j];//这里相当于计算那个num(456,456)
}
if(cnt==1)ans++;//这里也相当于计算那个num(456,456),但是个位数不会进入上面那个循环
return ans;
}
int main(){
scanf("%lld%lld%lld",&l,&r,&k);
dpf();
printf("%lld",num(r)-num(l-1));
return 0;
}
最新文章
- ZeroMQ接口函数之 :zmq_send_const – 从一个socket上发送一个固定内存数据
- Linux和UNIX监控
- Easyui Combotree问题及其相关
- phpcmsv9如何实现添加栏目时不在首页内容区显示只在导航栏显示
- [SSH 2] 以网站主页面浅谈Struts2配置
- 编写一函数用来实现左右循环移位。函数原型为move(value,n);n>;0时右移n位,n<;0时左移|n|位。
- PHP中长连接的实现
- 28.leetcode 13. Roman to Integer
- LGTB与序列 状压dp
- Java 容器 接口
- axure 预览";HTTP/1.1 302 Found";
- js文档碎片
- Java SQL注入学习笔记
- Excel的方向键失灵
- [转]连连看游戏 C#
- Java中 Random
- Mysql中datetime和timestamp区别
- Task WaitAll的用法
- Android性能优化之数据库优化
- iOS之点击通知栏跳转应用的相关页面