(hard还是有点强度的)
给定一个正整数 n
,请你统计在 [0, n]
范围的非负整数中,有多少个整数的二进制表示中不存在 连续的 1 。
示例 1:
输入: n = 5
输出: 5
解释:
下面列出范围在 [0, 5] 的非负整数与其对应的二进制表示:
0 : 0
1 : 1
2 : 10
3 : 11
4 : 100
5 : 101
其中,只有整数 3 违反规则(有两个连续的 1 ),其他 5 个满足规则。
示例 2:
输入: n = 1
输出: 2
示例 3:
输入: n = 2
输出: 3
提示:
1 <= n <= 109
数位DP或者 数学(找规律,本题是斐波那契数列)
数位DP介绍见:https://zhuanlan.zhihu.com/p/50791875
class Solution {
public:
int findIntegers(int n) {
int dp[32][2];
memset(dp,0,sizeof(dp));
dp[0][0]=1;
for(int i=1;i<32;i++)
{
dp[i][0]=dp[i-1][0]+dp[i-1][1]; //储存了当该位为1和为0时,非连续1数的个数
dp[i][1]=dp[i-1][0];
}
vector<int> record;
while(n)
{
int temp=n%2;
n/=2;
record.push_back(temp);
cout<<temp;
}
int ans=0;
int last=0;
int thisnum=0;
for(int i=record.size()-1;i>=0;i--)
{
thisnum=record[i];
if(thisnum)
{
ans+=dp[i+1][0];//注意由于dp和record起始不同,dp为1的是第一位,而record中i=1实际上是指示的第二位,因此计算第i位有多少符合情况的,应该记录dp[i+1][0]
if(thisnum==last) break; //发现前面已经存在两个连续1了,则低位那些符合情况的非连续1的情况也不能计算在内了,但是那两位的即01,10的情况还是要算在内的
}
last=thisnum;
if(i==0) ans+=1;
}
return ans;
}
};