代码随想录算法训练营第二十七天 | 56. 合并区间、738.单调递增的数字、968.监控二叉树

代码随想录算法训练营第二十七天任务

56. 合并区间

題目链接:56. 合并区间

思路:当遇到intervals[i][0] > end 将之前的start, end存储起来,更新start,end; 当重叠时,更新end.

cpp 复制代码
class Solution {
public:
    vector<vector<int>> merge(vector<vector<int>>& intervals) {
        sort(intervals.begin(), intervals.end());
        int start = intervals[0][0];
        int end = intervals[0][1];
        vector<vector<int>> res;
        for (int i = 1; i < intervals.size(); ++i) {
            if (intervals[i][0] > end) {
                res.push_back({start, end});
                start = intervals[i][0];
            }
            if (intervals[i][1] > end) {
                end = intervals[i][1];
            } 
        }
        res.push_back({start, end});   // 最后一个区间数组
        return res;
    }
};

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

空间复杂度:O(n) 排序

738.单调递增的数字

題目链接:738.单调递增的数字

看题解了,从后往前遍历。

让我想起了135. 分发糖果这道题从右向左遍历时的情景。

cpp 复制代码
class Solution {
public:
    int monotoneIncreasingDigits(int n) {
        string strNum = to_string(n);
        int flag = strNum.size();   // 记录哪一位是从 9 开始的. eg: 1000   不能初始化0或者-1,
        for (int i = strNum.size() - 1; i >= 1; --i) {
            if (strNum[i - 1] > strNum[i]) {   // 出现递减
                strNum[i - 1]--;
                strNum[i] = '9';
                flag = i;
            }
        }
        for (int i = flag + 1; i < strNum.size(); ++i) {
            strNum[i] = '9';
        }
        return atoi(strNum.c_str());
    }
};

时间复杂度:O(n)

空间复杂度:O(n) 字符串空间

968.监控二叉树

題目链接:968.监控二叉树

这道题相当困难。看题解了。

自己再写一下思路,加强理解。

首先是叶子节点肯定不放摄像头,这样能保证最少的摄像头数量。(叶子节点是指数级别的增长)

在层上,每隔两个放一个摄像头

树的遍历方式:后序遍历(左右后)

对于监控二叉树的节点,有三种状态:有摄像头,有覆盖,无覆盖。

1表示有摄像头

2表示有覆盖

3表示无覆盖

如果遇到空节点,肯定不是有摄像头,如果是无覆盖,那么叶子节点就要有摄像头(想象一个节点 左孩子 和 右孩子 只有其中一个),所以空节点的状态应该是有覆盖。

cpp 复制代码
/*
    用 1 表示有摄像头
    用 2 表示有覆盖
    用 3 表示无覆盖 : 叶子节点,父节点有摄像头
 */
class Solution {
    int result = 0;
    int traversal(TreeNode* cur) {
        if (cur == nullptr) return 2;  // 空节点表示有覆盖。
        int left = traversal(cur->left);
        int right = traversal(cur->right);

        // 左右节点都有覆盖
        if (left == 2 && right == 2) return 3;

        // 左右节点至少有一个是无覆盖,父节点就要放摄像头
        if (left == 3 || right == 3) {
            result++;
            return 1;
        }

        // 左右节点任一一个有摄像头, 父节点就是有覆盖
        if (left == 1 || right == 1) return 2;
        
        return -1;  // 逻辑不会走到这里,只是函数需要个返回值。
    }

public:
    int minCameraCover(TreeNode* root) {
        if (traversal(root) == 3) {   // 如果头节点无覆盖,就需要加摄像头
            result++;
        }
        return result;
    }
};

这道题有意思。

时间复杂度:O(n) 需遍历每个节点

空间复杂度:O(n) 递归栈深度

相关推荐
小poop2 小时前
string 类从入门到深入
c++
眠りたいです3 小时前
现代C++:C++14中的新语言特性和库特性
c语言·开发语言·c++
Black蜡笔小新3 小时前
自动化AI算法训练服务器DLTM助力医学影像分析进入AI智能分析新时代
人工智能·算法·自动化
手写码匠4 小时前
深入解析大模型架构之争:全能通用模型 vs 领域专精模型
人工智能·深度学习·算法·aigc
浅念-4 小时前
LeetCode 回溯算法题——综合练习
数据结构·c++·算法·leetcode·职场和发展·深度优先·dfs
列星随旋5 小时前
线段树和树状数组的学习
学习·算法
楼田莉子6 小时前
C++17新特性:__had_include/属性/求值顺序规则
开发语言·c++·后端
全糖可乐气泡水7 小时前
Codex适配国产信创环境安装部署与技术适配全解析
开发语言·git·python·算法·百度
h_a_o777oah7 小时前
状态机+划分型 DP :深度解析K-划分问题下 DP 状态的转移逻辑(洛谷P2679 P2331 附C++代码)
c++·算法·动态规划·acm·状态机dp·划分型dp·滚动数组优化