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

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

56. 合并区间

題目链接:56. 合并区间

思路:当遇到intervalsi0 > 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) 递归栈深度

相关推荐
老余捞鱼6 分钟前
线性回归实战:5步验证你的量化因子是否真有效
算法·金融·回归·线性回归·ai量化
charlie1145141918 分钟前
现代C++指南:Lambda,让我们用另一种方式持有函数
开发语言·c++
想吃火锅10059 分钟前
【leetcode】121.买卖股票的最佳时机js/c++
算法·leetcode·职场和发展
森G30 分钟前
77、线程池原理和实现------服务器源码解析----云视频服务项目
服务器·c++·qt
码云数智-大飞33 分钟前
RAII 与智能指针深度拆解
java·前端·算法
Dick50741 分钟前
ROS2 常用命令表
人工智能·学习·算法·机器人
.千余1 小时前
【C++】模板进阶全解:非类型参数|全特化|偏特化|分离编译完全指南
开发语言·c++·笔记·学习·其他
代码改善世界1 小时前
【C++进阶】C++11:列表初始化、右值引用与移动语义、完美转发全解析
java·开发语言·c++
apcipot_rain1 小时前
计科八股20260616(2)/面经——线性代数对称阵求n次幂、概率论最大似然估计
算法
牛油果子哥q1 小时前
并查集(DSU)超精讲,路径压缩、按秩合并、万能模板、连通性判定、最小生成树与刷题实战全解
数据结构·c++·最小生成树·并查集