600. 不含连续1的非负整数(数位dp或数学)

(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;
    }
};

 

posted @ 2022-11-22 21:50  zzzlight  阅读(50)  评论(0编辑  收藏  举报