【算法与数据结构】968、LeetCode监控二叉树

文章目录

  • 一、题目
  • 二、解法
  • 三、完整代码

所有的LeetCode题解索引,可以看这篇文章——【算法和数据结构】LeetCode题解。

一、题目

在这里插入图片描述
在这里插入图片描述

二、解法

  思路分析:本题的一共有两个难点,一个在于如何遍历二叉树(前中后遍历,选择什么遍历方式,迭代法或是递归法),另一个在于如何放置摄像头。首先观察例子,我们发现,为了寻找最小的摄像头数量,每个摄像头的辐射范围就要尽可能大,因此摄像头不会放在叶子节点上(少辐射一层节点)。因此我们会想到从叶子节点开始遍历,在其父节点放置摄像头。从层序遍历的视角来看就是从最底层开始遍历。注意这里不能在根节点的孩子节点放置摄像头,因为在叶子节点的父节点放置摄像头才能保证摄像头的数量最少。例如下图,三角形代表摄像头(在根节点的孩子节点放置摄像头不是最少的):
在这里插入图片描述
  因此,我们需要从叶子节点开始遍历,那么可以使用后序遍历(左右中遍历,【算法与数据结构】144、94、145LeetCode二叉树的前中后遍历(递归法、迭代法))。然后我们解决放置摄像头的问题。具体来说,节点的情况可以归纳为三种,无覆盖0,有摄像头1,有覆盖2。我们依据节点的返回值来确定是否放置摄像头,空节点必须归类为有覆盖这种情况,并返回2。如果左右节点都有覆盖(都返回2),则本节点没有覆盖,返回0。如左右节点有一个节点没有覆盖,则本节点需要放置一个摄像头,result++,返回1。如果左右节点至少有一个有摄像头,则本节点是有覆盖的状态,返回2。最后根节点根据左右孩子节点的返回值确定是否需要放置摄像头。
  程序如下

class Solution {
private:
    int result = 0; 
    // 节点一共三种情况,无覆盖0,有摄像头1,有覆盖2
    int traversal_postOrder(TreeNode* cur) {
        if (cur == NULL) return 2; // 空节点返回有覆盖

        int left = traversal_postOrder(cur->left);     // 左
        int right = traversal_postOrder(cur->right);    // 右
        
        // 情况一: 左右节点都有覆盖,本节点则无覆盖返回0
        if (left == 2 && right == 2) return 0;

        // 情况二:左右节点有一个节点没有覆盖
        if (left == 0 || right == 0) {
            result++;
            return 1;
        }

        // 情况三:左右节点至少有一个有摄像头
        if (left == 1 || right == 1) return 2;

        return -1;      // 所有情况已在上面,不会走到这里,只是满足编译器需要
    }

public:
    int minCameraCover(TreeNode* root) {
        // 1.后序(左右中)遍历,即从叶子节点开始 2.判断摄像头覆盖情况:一共有三种情况,不同情况返回不同的值
        if (traversal_postOrder(root) == 0) {
            result++;
        }
        return result;
    }
};

复杂度分析:

  • 时间复杂度: O ( n ) O(n) O(n),遍历二叉树中的每个节点。
  • 空间复杂度: O ( n ) O(n) O(n),存储整个二叉树。

  简化版本程序如下

class Solution {
private:
    int result = 0; 
    int traversal_postOrder(TreeNode* cur) {
        if (cur == NULL) return 2; // 空节点返回有覆盖
        int left = traversal_postOrder(cur->left);     // 左
        int right = traversal_postOrder(cur->right);    // 右       
        if (left == 2 && right == 2) return 0;  // 情况一: 左右节点都有覆盖,本节点则无覆盖返回0
        else if(left == 0 || right == 0) {      // 情况二:左右节点有一个节点没有覆盖
            result++;
            return 1;
        }else return 2;                         // 情况三:左右节点至少有一个有摄像头
    }

public:
    int minCameraCover(TreeNode* root) {
        if (traversal_postOrder(root) == 0) result++;
        return result;
    }
};

三、完整代码

# include <iostream>
# include <vector>
# include <stack>
# include <string>
# include <queue>
using namespace std;

// 树节点定义
struct TreeNode {
    int val;
    TreeNode* left;
    TreeNode* right;
    TreeNode() : val(0), left(nullptr), right(nullptr) {}
    TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
    TreeNode(int x, TreeNode* left, TreeNode* right) : val(x), left(left), right(right) {}
};

// 前序遍历迭代法创建二叉树,每次迭代将容器首元素弹出(弹出代码还可以再优化)
void Tree_Generator(vector<string>& t, TreeNode*& node) {
    if (!t.size() || t[0] == "NULL") return;    // 退出条件
    else {
        node = new TreeNode(stoi(t[0].c_str()));    // 中
        if (t.size()) {
            t.assign(t.begin() + 1, t.end());
            Tree_Generator(t, node->left);              // 左
        }
        if (t.size()) {
            t.assign(t.begin() + 1, t.end());
            Tree_Generator(t, node->right);             // 右
        }
    }
}

/*************************层序遍历****************************/ 
vector<vector<int>> levelOrder(TreeNode* root) {
    queue<TreeNode*> que;
    if (root != NULL) que.push(root);
    vector<vector<int>> result;
    while (!que.empty()) {
        int size = que.size();  // size必须固定, que.size()是不断变化的
        vector<int> vec;
        for (int i = 0; i < size; ++i) {
            TreeNode* node = que.front();
            que.pop();
            vec.push_back(node->val);
            if (node->left) que.push(node->left);
            if (node->right) que.push(node->right);
        }
        result.push_back(vec);
    }
    return result;
}
/****************************************************************/

/*******************************打印函数****************************/
template<typename T>
void my_print(T& v, const string msg)
{
    cout << msg << endl;
    for (class T::iterator it = v.begin(); it != v.end(); it++) {
        cout << *it << ' ';
    }
    cout << endl;
}
template<class T1, class T2>
void my_print2(T1& v, const string str) {
    cout << str << endl;
    for (class T1::iterator vit = v.begin(); vit < v.end(); ++vit) {
        for (class T2::iterator it = (*vit).begin(); it < (*vit).end(); ++it) {
            cout << *it << ' ';
        }
        cout << endl;
    }
}
/****************************************************************/

class Solution {
private:
    int result = 0; 
     节点一共三种情况,无覆盖0,有摄像头1,有覆盖2
    //int traversal_postOrder(TreeNode* cur) {
    //    if (cur == NULL) return 2; // 空节点返回有覆盖

    //    int left = traversal_postOrder(cur->left);     // 左
    //    int right = traversal_postOrder(cur->right);    // 右
    //    
    //    // 情况一: 左右节点都有覆盖,本节点则无覆盖返回0
    //    if (left == 2 && right == 2) return 0;

    //    // 情况二:左右节点有一个节点没有覆盖
    //    if (left == 0 || right == 0) {
    //        result++;
    //        return 1;
    //    }

    //    // 情况三:左右节点至少有一个有摄像头
    //    if (left == 1 || right == 1) return 2;

    //    return -1;      // 所有情况已在上面,不会走到这里,只是满足编译器需要
    //}
    int traversal_postOrder(TreeNode* cur) {
        if (cur == NULL) return 2; // 空节点返回有覆盖
        int left = traversal_postOrder(cur->left);     // 左
        int right = traversal_postOrder(cur->right);    // 右       
        if (left == 2 && right == 2) return 0;  // 情况一: 左右节点都有覆盖,本节点则无覆盖返回0
        else if(left == 0 || right == 0) {      // 情况二:左右节点有一个节点没有覆盖
            result++;
            return 1;
        }else return 2;                         // 情况三:左右节点至少有一个有摄像头
    }

public:
    int minCameraCover(TreeNode* root) {
        // 1.后序(左右中)遍历,即从叶子节点开始 2.判断摄像头覆盖情况:一共有三种情况,不同情况返回不同的值
        if (traversal_postOrder(root) == 0) result++;
        return result;
    }
};

int main() {
    // 构建二叉树
    vector<string> t = { "0", "0", "0", "NULL", "NULL", "0", "NULL", "NULL", "NULL" };   // 前序遍历,中左右
    my_print(t, "目标树前序遍历");
    TreeNode* root = new TreeNode();
    Tree_Generator(t, root);
    vector<vector<int>> tree = levelOrder(root);
    my_print2<vector<vector<int>>, vector<int>>(tree, "目标树层序遍历:");

    Solution s;
    int result = s.minCameraCover(root);
    cout << "所需摄像头数量:" << result << endl;

    system("pause");
    return 0;
}

end

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/286645.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

MySQL 时间日期函数,流程控制函数,加密解密函数以及聚合查询函数

注:本文仅作为查找函数和部分理解使用,希望能给大家带来帮助 以下函数均可以使用 SELECT NOW()等函数 FROM DUAL;来测试 //其中dual是一个准们用来测试的测试表 1.时间日期函数 1.1 获取时间的函数 重点记忆前三个红色标注的函数, 第一个函数返回值如2024-01-02的形式 第二个如…

如何使用curl在PHP中同时上传文件和其他数据?

问CHAT&#xff1a;举个例子说明如何使用curl在PHP中同时上传文件和其他数据&#xff1f; CHAT回复&#xff1a;以下例子为&#xff1a; php <?php $url http://www.example.com/path/; $filename path/to/your/file.png; $fields array( fieldParam1 > someValue, …

C++摸版(初阶)----函数模版与类模版

本专栏内容为&#xff1a;C学习专栏&#xff0c;分为初阶和进阶两部分。 通过本专栏的深入学习&#xff0c;你可以了解并掌握C。 &#x1f493;博主csdn个人主页&#xff1a;小小unicorn ⏩专栏分类&#xff1a;C &#x1f69a;代码仓库&#xff1a;小小unicorn的代码仓库&…

Mac环境下反编译apk

Mac环境下反编译apk 安装反编译工具dex2jar&#xff1a;[官网下载](https://sourceforge.net/projects/dex2jar/)JD-GUI&#xff1a;[官网下载](https://jd-gui.apponic.com/) 实操1. 将需要反编译的 .apk 文件放在下载的 dex2jar 文件夹目录下2. 使用 cd /xxx/dex2jar-2.0 命令…

深度生成模型之GAN的评估 ->(个人学习记录笔记)

文章目录 深度生成模型之GAN的评估图像翻译的应用1. 风格迁移2. 数据增强3. 经典图像任务4. 内容创作5. 人脸图像编辑6. 人体图像编辑 图像翻译模型1. 有监督图像翻译模型2. 无监督图像翻译模型3. 多域图像翻译模型 深度生成模型之GAN的评估 图像翻译的应用 1. 风格迁移 各类…

120基于matlab的LMS自适应滤波算法

基于matlab的LMS自适应滤波算法&#xff0c;如、解相关LMS算法&#xff0c;滤波型LMS算法&#xff0c;变换域LMS算法&#xff0c;输出滤波前后及学习曲线图。数据可更换自己的&#xff0c;程序已调通&#xff0c;可直接运行。 120自适应滤波算法变换域LMS算法 (xiaohongshu.com…

k8s的陈述式管理

k8s的陈述式管理&#xff1a; 所谓的陈述式管离也就是命令行工具 优点&#xff1a;90%以上都可以满足 对资源的增删查比较方便&#xff0c;对改不是很友好 缺点&#xff1a;命令比较冗长&#xff0c;复杂&#xff0c;难记 声明式&#xff1a; k8s当中的YAML文件来实现资源管…

测试要不要写测试用例(多方面剖析)

前言 最近在网上看到关于这样的一个话题“测试需要写测试用例吗&#xff1f;”&#xff0c;引起了很多同学的讨论。这段话是这样的&#xff1a; 测试用例主要作用&#xff1a;有效地评估软件的质量&#xff0c;测试用例质量体现了测试的质量。 下面摘取一部分同学的观点&…

对比开源大语言模型的自然语言生成SQL能力

背景 NL-to-SQL&#xff08;自然语言到结构化查询语言&#xff09;任务是自然语言处理&#xff08;NLP&#xff09;领域的一个难题。 它涉及将自然语言问题转换为 SQL 查询&#xff0c;然后可以针对关系数据库执行该查询来回答问题。 该任务是 NLP 中的一个专门子领域&#xf…

Java企业电子招投标系统源代码,支持二次开发,采用Spring cloud框架

在数字化采购领域&#xff0c;企业需要一个高效、透明和规范的管理系统。通过采用Spring Cloud、Spring Boot2、Mybatis等先进技术&#xff0c;我们打造了全过程数字化采购管理平台。该平台具备内外协同的能力&#xff0c;通过待办消息、招标公告、中标公告和信息发布等功能模块…

SQL 在已有表中修改列名的方法

文章目录 1. MySQL2. SQL Server3. Oracle / PostgreSQL Question&#xff1a; 假设有一张表 StudentInfo&#xff0c;表中有一个列名是 Student_Name &#xff0c;想要把这个列名改成 StudentName 应该如何操作&#xff1f; 建表语句如下&#xff1a; --建表 if object_id(S…

Java-网络爬虫(一)

文章目录 前言一、网络爬虫1. 介绍2. 爬虫协议3. 法律法规 二、相关知识1. HttpClient2. Jsoup 三、综合案例1. 案例一2. 案例二 四、总结 前言 在大数据时代&#xff0c;信息采集是一项重要的工作&#xff0c;而互联网中的数据是海量的&#xff0c;如果单纯靠人力进行信息获取…

MATLAB习题操作实战

2.1创建一个有7个元素的一维数组&#xff0c;并做如下处理:直接寻访一维数组的第6个元素;寻访一维数组的第1、3、5个元素;寻访一维数组中第4个至最后1个元素;寻访一维数组中大于70的元素。 % 创建一维数组 array [50, 60, 70, 80, 90, 100, 110];% 直接寻访一维数组的第6个元…

【自动驾驶中的SLAM技术】第2讲:基础数学知识回顾

第二讲&#xff1a;基础数学回顾 文章目录 第二讲&#xff1a;基础数学回顾1 几何学1.1 坐标系1.2 坐标变换① 空间向量② 基变换③ 坐标变换④ 总结 1.3 四元数与旋转向量 2 运动学2.1 李群视角2.2 四元数视角2.3 四元数的李代数与旋转向量间的转换2.4 SO(3)t 上的运动学2.5 线…

六、HTML 段落

HTML 可以将文档分割为若干段落。 一、HTML 段落 段落是通过 <p> 标签定义的。 <p>这是一个段落 </p> <p>这是另一个段落</p> 注意&#xff1a;浏览器会自动地在段落的前后添加空行。&#xff08;</p> 是块级元素&#xff09; 二、不…

算法巡练day03Leetcode203移除链表元素707设计链表206反转链表

今日学习的文章视频链接 https://www.bilibili.com/video/BV1nB4y1i7eL/?vd_source8272bd48fee17396a4a1746c256ab0ae https://programmercarl.com/0707.%E8%AE%BE%E8%AE%A1%E9%93%BE%E8%A1%A8.html#%E7%AE%97%E6%B3%95%E5%85%AC%E5%BC%80%E8%AF%BE 链表理论基础 见我的博…

Linux 命令echo

命令作用 输出一行字符串在shell中&#xff0c;可以打印变量的值输出结果写入到文件在显示器上显示一段文字&#xff0c;起到提示的作用 语法 echo [选项] [字符串] 参数 字符含义-n不自动换行-e解释转义字符-E不解释转义字符 如果-e有效&#xff0c;则识别以下序列&…

2024,这将是量子计算的真正挑战

2023年&#xff0c;一项项量子计算纪录被打破。 谷歌量子AI团队证明了将多个量子比特分组合成为一个逻辑量子比特的纠错方法可以提供更低的容错率。以往的纠错研究随着比特数的增加&#xff0c;错误率会提高&#xff0c;都是“越纠越错”&#xff0c;而这次谷歌首次实现了“越纠…

K8S本地开发环境-minikube安装部署及实践

引言 在上一篇介绍了k8s的入门和实战&#xff0c;本章就来介绍一下在windows环境如何使用minikube搭建K8s集群&#xff0c;好了废话不多说&#xff0c;下面就和我一起了解Minikube吧。 什么是Minikube&#xff1f; Minikube 是一种轻量级的 Kubernetes 实现&#xff0c;可在本…

1688商品详情API:实现商品详情自动化的关键步骤

一、准备工作 在使用1688商品详情API之前&#xff0c;我们需要进行一些准备工作。 注册与登录&#xff1a;首先&#xff0c;你需要在1688的开放平台上注册一个账号并创建一个应用。这样你就可以获得一个API密钥&#xff0c;这是调用API的凭证。阅读API文档&#xff1a;详细阅…