标签:含义 10个 out return hint void efi ++ sig
1 99
9 20 20 20 20 20 20 20 20 20
30%的数据中,a<=b<=10^6;
100%的数据中,a<=b<=10^12。
十进制数位dp
设f(i,j,k)表示第i位以j开头的数含有多少个数字k
先预处理出f[i][j][k]数组
转移的时候从高到低转移
#include<stdio.h> #define LL unsigned long long int 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)
标签:含义 10个 out return hint void efi ++ sig
原文地址:http://www.cnblogs.com/keshuqi/p/6281398.html