数位DP——不用62和4 ( HDU 2089 )
来源:互联网 发布:戴威尔网络 编辑:程序博客网 时间:2024/06/17 07:20
题目链接:
http://acm.hdu.edu.cn/showproblem.php?pid=2089分析:
给出啊一个范围[n, m],求范围内不含62与4的数字个数。题解①:
- 定义dp[i][j]为长度为i ,最高位为j的不含62与4的数字的个数。
- DP代码:
void init(){ memset(dp,0,sizeof(dp)); int i,j,k; dp[0][0]=1; for(i=1; i<=7; i++)//数字长度 { for(j=0; j<10; j++)//长度为i的数字最高位为i { for(k=0; k<10; k++)//长度为i-1的数字最高位为j { if(j!=4 &&!(j==6 &&k==2))//如果j不等于4并且j和k不构成62 dp[i][j]+=dp[i-1][k]; } } }}
solve代码:
for(i=len; i; i--)//遍历数字长度{ for(j=0; j<num[i]; j++)当数字长度为i时后,遍历此时最高位0~num[i] { if(j!=4 &&!(num[i+1]==6 &&j==2))//如果不含4并且和上一位不构成64那么加上dp[i][j] ans+=dp[i][j]; } if(num[i]==4 || (num[i+1]==6 &&num[i]==2)) break; cout << len<< "==" <<ans << endl;}return ans;
- 题解②:
- 定义dp[i][j]数组;
- dp[i][0]表示长度为i,不含有4也不含有62子串最高位也不为2的数字个数;
- dp[i][1]表示长度为i,不含有4也不含有62子串但是最高位为2的数字个数;
- dp[i][2]表示长度为i,含有4或者含有64子串的数字个数;
- DP代码:
memset(dp,0, sizeof(dp)); dp[0][0]=1;for (int i = 1; i < 10; ++i){ dp[i][0] = dp[i-1][0] * 9 - dp[i-1][1];//dp[i][0]等于dp[i-1][0]乘以9(不包括4在内)减去dp[i-1][1](i-1长度最高位为2数字个数)。 dp[i][1] = dp[i-1][0];//就等于i-1长度时候什么都不含的数字个数,第i位为2。 dp[i][2] = dp[i-1][2] * 10 + dp[i-1][0] + dp[i-1][1];//等于i-1长度时候满足条件的数字个数*10倍,再加上i-1长度什么都不含(i位为4的就满足)和i-1长度最高位为2(i位为6就满足)}
solve代码:
long long ans = 0;bool flag = false;for (int i = len; i >0; --i) { ans += (dp[i-1][2] * a[i]); if (flag) ans += dp[i-1][0] * a[i]; if (!flag && a[i] > 4) ans += dp[i-1][0]; if(!flag&& a[i]>6) ans += dp[i-1][1]; if(!flag&&a[i+1]==6&&a[i]>2) ans+=dp[i][1]; if ((a[i+1] == 6 && a[i] == 2)||a[i]==4) flag = true;}
- 解法①AC代码:
#include <cstdio>#include <algorithm>#include <cstring>#include <cmath>#include <iostream>#include <queue>#include <stack>using namespace std;int dp[12][10];int num[12];void init(){ memset(dp,0,sizeof(dp)); int i,j,k; dp[0][0]=1; for(i=1; i<=7; i++) { for(j=0; j<10; j++) { for(k=0; k<10; k++) { if(j!=4 &&!(j==6 &&k==2)) dp[i][j]+=dp[i-1][k]; } } }}int solve(int a){ int i,j; int len=0; int ans=0; while(a) { num[++len]=a%10; a/=10; } num[len+1]=0; for(i=len; i; i--)//遍历数字长度 { for(j=0; j<num[i]; j++)当数字长度为i时后,遍历此时最高位0~num[i] { if(j!=4 &&!(num[i+1]==6 &&j==2))//如果不含4并且和上一位不构成64那么加上dp[i][j] ans+=dp[i][j]; } if(num[i]==4 || (num[i+1]==6 &&num[i]==2)) break; cout << len<< "==" <<ans << endl; } return ans;}int main(){ init(); int a,b; while(scanf("%d%d",&a,&b)!=EOF&&a+b!=0) { int res=solve(b+1)-solve(a); cout<<res<<endl; } return 0;}
- 解法②AC代码:
#include<iostream>#include<cstring>using namespace std;long long dp[10][3];long long n;int a[10];long long solve(long long n){ long long sum=n; int len = 0; memset(a, 0, sizeof(a)); while (n) { a[++len] = n % 10; n /= 10; } long long ans = 0; bool flag = false; for (int i = len; i >0; --i) { ans += (dp[i-1][2] * a[i]); if (flag) ans += dp[i-1][0] * a[i]; if (!flag && a[i] > 4) ans += dp[i-1][0]; if(!flag&& a[i]>6) ans += dp[i-1][1]; if(!flag&&a[i+1]==6&&a[i]>2) ans+=dp[i][1]; if ((a[i+1] == 6 && a[i] == 2)||a[i]==4) flag = true; } return sum-ans;}int main(){ memset(dp,0, sizeof(dp)); dp[0][0]=1; for (int i = 1; i < 10; ++i) { dp[i][0] = dp[i-1][0] * 9 - dp[i-1][1]; dp[i][1] = dp[i-1][0]; dp[i][2] = dp[i-1][2] * 10 + dp[i-1][0] + dp[i-1][1]; } long long l,r; while (cin>>l>>r&&l+r) { cout<<solve(r+1)-solve(l)<<endl; } return 0;}
0 0
- 数位DP——不用62和4 ( HDU 2089 )
- HDU 2089——不要62(数位DP)
- hdu(2089)——不要62(数位dp)
- HDU 2089 不要62——数位dp
- [数位DP]Hdu 2089——不要62
- HDU 2089 数位DP
- Hdu 2089 数位dp
- HDU 2089 数位DP
- 数位DP hdu-2089
- HDU 2089 数位DP
- HDU 2089【数位DP】
- 【数位DP】 HDU 2089
- 数位DP hdu 2089
- hdu 2089 数位DP
- hdu 2089 数位dp
- hdu 2089 数位dp
- hdu 2089 数位dp
- HDU 2089 数位DP
- Linux进程模型2
- STM32F401 SDIO HAL driver初始化SD卡
- 存在*.lastUpdated文件也能更新jar文件的方法
- 我的博客学习第一步
- 视频---码率、帧率、清晰度、分辨率、流畅度的关系与设置
- 数位DP——不用62和4 ( HDU 2089 )
- 为什么无穷大总是0x3f3f3f3f?
- CCNA系列二之RIP路由
- 【经验分享】如何将LPCXpresso的仿真器烧录成CMSIS-DAP或JLINK?
- APUE读书笔记-第八章 进程控制
- faster rcnn源码解读总结
- 国际化
- 快速修复网页在IE8 下的显示兼容问题
- java的二维数组