数位DP LeetCode 600 不含连续1的非负整数

一、题目

1、题目描述

给定一个正整数 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、接口描述

复制代码
cpp 复制代码
class Solution {
public:
    int findIntegers(int n) {

    }
};

3、原题链接

600. 不含连续1的非负整数


二、解题报告

1、思路分析

本题是数位DP的模板题,关于数位DP笔者将于明日发布详细讲解,这里仅给出本题解法

我们设计状态f[i][j]为剩余i位未处理,上一位为j情况下目标数字的数目,这也是数位dp问题通用状态设计

那么我们只需要枚举下一位即可,如果前面的位严格小于给定数字的对应位,那么可以从0枚举到9,否则只能枚举到给定数字对应位

对于连续1的情况不进行枚举即可

2、复杂度

时间复杂度: O(log n) 空间复杂度:O(log n)

3、代码详解

复制代码
cpp 复制代码
int f[32][32] , d[32] , cnt = 0;
class Solution {
public:
    Solution()
    {memset(f , -1 , sizeof(f));}
    int dfs(int n , int pre , bool lim)
    {
        if(!n) return 1;
        if(lim && ~f[n][pre]) return f[n][pre];
        int ceil = lim ? 1 : d[n];
        int res = 0;
        for(int i = 0 ; i <= ceil ; i++)
            if(pre == 1 && i == 1) continue;
            else res += dfs(n - 1 , i , lim || i < ceil);
        if(lim)
        f[n][pre] = res;
        return res;
    }
    int getnum(int x)
    {
        cnt = 0;
        while(x) d[++cnt] = x % 2 , x /= 2;
        return dfs(cnt , 0 , false);
    }
    int findIntegers(int n) {
        return getnum(n);
    }
};
相关推荐
hxung3 小时前
B+树与红黑树
数据结构·b树
龙俊杰的读书笔记4 小时前
[leetcode] 面试经典 150 题——篇9:二叉树(番外:二叉树的遍历方式)
数据结构·算法·leetcode·面试
刚入门的大一新生4 小时前
C++初阶-C++的讲解1
开发语言·c++
Swift社区5 小时前
从表格到序列:Swift 如何优雅地解 LeetCode 251 展开二维向量
开发语言·leetcode·swift
sml259(劳改版)6 小时前
数据结构--堆
数据结构·算法·
ALex_zry7 小时前
C++17模板编程与if constexpr深度解析
开发语言·c++·性能优化
独家回忆3647 小时前
每日算法-250409
算法
菜还不练就废了7 小时前
4.5蓝桥杯|高塔登顶方案(5025)
职场和发展·蓝桥杯
青椒大仙KI118 小时前
25/4/6 算法笔记<仿真O2DES>基础知识学习
笔记·学习·算法
井云智能AI矩阵系统8 小时前
数字人情感表达突破:微表情自动生成的算法革新
算法