Valley Numer
Accepts: 548
Submissions: 1125
Time Limit: 2000/1000 MS (Java/Others)
Memory Limit: 32768/32768 K (Java/Others)
題目鏈接:http://bestcoder.hdu.edu.cn/contests/contest_showproblem.php?cid=777&pid=1005
題目很容易就能看出來這個一個數位dp,比較麻煩,但是還好,難度一般。
經典的數位DP,可以將狀態設計成四維,這裏我用的三維。
當前數字長度len ,最後一位數字digit,是否已經在遞增序列裏increased 是否和當前前綴相同same_prefix
轉移時處理好這些狀態就好了。
代碼:
#include <cstdio>
#include <cstring>
#include <iostream>
#define LL long long
using namespace std;
const LL mod=1e9+7;
char s[200];
int d[200],len;
LL dp[200][20][20];
LL dfs(int pos,bool flag,bool first,int now,int pre){
if(pos==-1)
return 1;
if(!flag&&!first&&~dp[pos][now][pre]){
return dp[pos][now][pre];
}
LL ans=0;
int limit=flag?d[pos]:9;
for(int i=0;i<=limit;i++){
if(i==0&&first){
ans+=dfs(pos-1,flag&&i==limit,first,now,pre);
ans%=mod;
}
else{
if(now>pre){
if(i<now)
continue;
}
if(now==i)
ans+=dfs(pos-1,flag&&i==limit,first&&i==0,i,pre);
else
ans+=dfs(pos-1,flag&&i==limit,first&&i==0,i,now);
ans%=mod;
}
}
if(!flag&&!first)
dp[pos][now][pre]=ans;
return ans;
}
int main(){
int t;
scanf("%d",&t);
memset(dp,-1,sizeof(dp));
while(t--){
scanf("%s",s);
len=strlen(s);
for(int i=0;i<len;i++){
d[i]=s[i]-'0';
}
for(int i=0;i<len/2;i++){
swap(d[i],d[len-1-i]);
}
LL ans=dfs(len-1,1,1,10,10)-1;
ans=(ans%mod+mod)%mod;
cout<<ans<<endl;
}
}