标签:
题意:
A数列为,包含7或者能被7整除的数从小到大构成。
{a[1]=7,a[2]=14,a[3]=17,a[4]=21,a[5]=27,a[6]=28,a[7]=35,a[8]=37,a[9]=42,a[10]=47};
B数列为,是A数列里的数,但不包含第a[i]个A数列里的数列。
也就是B数列要去掉 a[7],a[14],a[17]这些要去掉。
{b[1]=7,b[2]=14,b[3]=17,b[4]=21,b[5]=27,b[6]=28,b[7]=37,b[8]=42,b[9]=47,b[10]=49};
现在给N,求第N个B数列里的数。
思路:
首先值得肯定的是,我们必须知道A数列,所以数位dp关于A数列。这个都很简单。
dp[i][j][k] 第i位,余数是j,k 代表是否含有了7.
然后就看第N个B数列的数,实际是第几个A数列的数。
我们假设是第N个B 数列的数,是第X个A数列的数。
那么满足N=X-solve(X) solve(X) 代表1~X中有几个A数列的数。
会发现X-solve(X) 随着X的增大而增大。
所以二分出最小满足X-solve(X)=N 的X。
接着再一个二分求第X个A数列的数是多少就可以了~
最后就是需要注意,题目要求范围是2^63-1,所以开成 unsigned long long。
代码:
#include"stdio.h" #include"algorithm" #include"string.h" #include"map" #include"iostream" #include"queue" #include"string" #define ll unsigned long long using namespace std; ll dp[22][8][2]; int num[22]; ll dfs(int site,int s,int o,int f) { if(site==0) { if(o==1) return 1; if(s==0) return 1; return 0; } if(!f && dp[site][s][o]!=-1) return dp[site][s][o]; ll ans=0; int len=f?num[site]:9; for(int i=0; i<=len; i++) { if(o==1) ans+=dfs(site-1,(s*10+i)%7,1,f&&i==len); else { if(i==7) ans+=dfs(site-1,(s*10+i)%7,1,f&&i==len); else ans+=dfs(site-1,(s*10+i)%7,0,f&&i==len); } } if(!f) dp[site][s][o]=ans; return ans; } ll solve(ll x) { int cnt=0; while(x) { num[++cnt]=x%10; x/=10; } return dfs(cnt,0,0,1)-1; } int main() { ll n; memset(dp,-1,sizeof(dp)); while(scanf("%llu",&n)!=-1) { ll l=1,r=(1LL<<63)-1,mid,kx; while(l<=r) { mid=(l+r)/2; ll tep=mid-solve(mid); if(tep>=n) { kx=mid; r=mid-1; } else l=mid+1; } ll ans; l=1,r=(1LL<<63)-1; while(l<=r) { mid=(l+r)/2; if(solve(mid)>=kx) { ans=mid; r=mid-1; } else l=mid+1; } printf("%llu\n",ans); } return 0; }
[数位dp] upcoj 2223 A-Number and B-Number
标签:
原文地址:http://blog.csdn.net/wdcjdtc/article/details/44963259