首页 > 代码库 > [bzoj1833][ZJOI2010][count] (数位dp)
[bzoj1833][ZJOI2010][count] (数位dp)
Description
给定两个正整数a和b,求在[a,b]中的所有整数中,每个数码(digit)各出现了多少次。
Input
输入文件中仅包含一行两个整数a、b,含义如上所述。
Output
输出文件中包含一行10个整数,分别表示0-9在[a,b]中出现了多少次。
Sample Input
1 99
Sample Output
9 20 20 20 20 20 20 20 20 20
HINT
30%的数据中,a<=b<=10^6;
100%的数据中,a<=b<=10^12。
Solution
十进制数位dp
设f(i,j,k)表示第i位以j开头的数含有多少个数字k
先预处理出f[i][j][k]数组
转移的时候从高到低转移
#include<stdio.h>#define LL unsigned long longint len,zt[33];LL f[33][33][33],bt[33],a,b,ans[33];void init() { bt[1]=1; for(int i=2; i<=13; i++) bt[i]=bt[i-1]*10; for(int i=0; i<10; i++) f[1][i][i]=1; for(int i=2; i<=13; i++) for(int j=0; j<10; j++) for(int k=0; k<10; k++) { for(int l=0; l<10; l++) f[i][j][l]+=f[i-1][k][l]; f[i][k][k]+=bt[i-1]; } }void solve(LL x,LL d) { LL bef=x; for(len=0; x; x/=10) zt[++len]=x%10; for(int i=1; i<len; i++) for(int j=1; j<10; j++) for(int k=0; k<10; k++) ans[k]+=d*f[i][j][k]; for(int i=len; i; i--) { for(int j=0; j<zt[i]; j++) { if(!j && i==len)continue; for(int k=0; k<10; k++) ans[k]+=d*f[i][j][k]; } ans[zt[i]]+=d*(bef%bt[i]+1); } }int main() { init(); scanf("%lld%lld",&a,&b); solve(b,1); solve(a-1,-1); for(int i=0; i<9; i++) printf("%lld ",ans[i]); printf("%lld\n",ans[9]); return 0; }
[bzoj1833][ZJOI2010][count] (数位dp)
声明:以上内容来自用户投稿及互联网公开渠道收集整理发布,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任,若内容有误或涉及侵权可进行投诉: 投诉/举报 工作人员会在5个工作日内联系你,一经查实,本站将立刻删除涉嫌侵权内容。