0一、html
有個易錯點,我卡了很久,就是 1(012)#(1) 答案是1012,我剛開始把012的前導0去掉了。。。spa
0三、code
原題htm
0五、blog
數位DP,記憶化搜索遞歸
dp[ p ][ q ][ limit ][ ceng ]:表示前一位是 p,前面處於遞增(遞減)狀態q,當前位上限是多少limit,還有多少位ceng .get
根據不一樣狀況遞歸便可,能夠先不寫記憶化,先寫暴力的,最後加上記憶化。string
1 #include<stdio.h> 2 #include<string.h> 3 #define Mod 1000000007 4 #define LL long long 5 LL dp[105][11][11][2]; 6 7 LL dfs(int p, int q, int limit, int ceng){ 8 if(ceng==0){ 9 return 1; 10 } 11 if(dp[ceng][p+1][limit][q]!=-1) return dp[ceng][p+1][limit][q]; 12 13 LL ret = 0; 14 for(int i=0; i<limit; i++){ 15 if(q==0){ // 遞減 16 if(p==-1 || i <= p) { 17 if(p==-1 && i==0){ 18 ret += dfs(-1, 0, 10, ceng-1); 19 } else 20 ret += dfs(i, 0, 10, ceng-1); 21 } else { 22 ret += dfs(i, 1, 10, ceng-1); 23 } 24 } else { 25 if(i < p) continue; 26 ret += dfs(i, 1, 10, ceng-1); 27 } 28 ret %= Mod; 29 } 30 31 dp[ceng][p+1][limit][q]=ret; 32 return ret; 33 } 34 35 int main(){ 36 memset(dp, -1, sizeof(dp)); 37 char n[105]; 38 int t; 39 scanf("%d", &t); 40 while(t--){ 41 scanf("%s", &n); 42 LL ans = 0; 43 int len = strlen(n); 44 int u = 0; 45 for(int i=0; i<len; i++){ 46 int limit = n[i]-'0'; 47 int p = i==0?-1:n[i-1]-'0'; 48 49 if(u==2) break; 50 51 ans += dfs(p, u, limit, len-i); 52 ans %= Mod; 53 54 if(u==0){ 55 if(i!=0 && limit>p){ 56 u=1; 57 } 58 } else if(u==1){ 59 if(limit<p){ 60 u=2; 61 } 62 } 63 } 64 printf("%I64d\n", ( (ans - 1 + (u==2?0:1) ) % Mod + Mod ) % Mod ); 65 } 66 return 0; 67 }