此题新颖之处在于二进制的数位dp,平常见的都是十进制数位dp
因为要统计0的数量,所以前导零会有影响,
那么当dp也只在没有前导零的时候才记忆化。
抄自洛谷:
由于我们要搜的数可能很长,所以我们的直接最高位搜起
举个例子:假如我们要从 [0,1000] 找任意相邻两数相等的数
显然 111,222,888 等等是符合题意的数
但是我们发现右端点 1000 是四位数
因此我们搜索的起点是 0000 ,而三位数的记录都是 0111,0222,0888 等等
而这种情况下如果我们直接找相邻位相等则 0000 符合题意而 0111,0222,0888 都不符合题意了
所以我们要加一个前导0标记
如果当前位 lead=1 而且当前位也是0,那么当前位也是前导0, pos+1 继续搜;
如果当前位 lead=1 但当前位不是0,则本位作为当前数的最高位, pos+1 继续搜;(注意这次根据题意st或其他参数可能发生变化)
#include<bits/stdc++.h> using namespace std; int a[32]; int dp[32][64];//0-1,base=32 int dfs(int pos,int sta,bool limit,bool lead) { if(pos==-1) return sta>=32; if(!lead && !limit && dp[pos][sta]!=-1) return dp[pos][sta]; int res=0; for(int i=0;i<=(limit?a[pos]:1);++i) { if(lead && i==0) res+=dfs(pos-1,sta,limit&&i==a[pos],lead); else res+=dfs(pos-1,sta+(i==0?1:-1),limit&&i==a[pos],lead&&i==0); } if(!lead && !limit) dp[pos][sta]=res; return res; } int solve(int x) { int pos=0; while(x) { a[pos++]=x&1; x>>=1; } return dfs(pos-1,32,true,true); } int main() { memset(dp,-1,sizeof(dp)); int le,ri; while(~scanf("%d%d",&le,&ri)) { printf("%d\n",solve(ri)-solve(le-1)); } return 0; }