【二叉树的深搜】二叉树剪枝

文章目录

  • 814. 二叉树剪枝
  • 解题思路:深度优先遍历 + 后序遍历
  • 另一种写法

在这里插入图片描述

814. 二叉树剪枝

814. 二叉树剪枝

​ 给你二叉树的根结点 root ,此外树的每个结点的值要么是 0 ,要么是 1

​ 返回移除了所有不包含 1 的子树的原二叉树。

​ 节点 node 的子树为 node 本身加上所有 node 的后代。

示例 1:

img
输入:root = [1,null,0,0,1]
输出:[1,null,0,null,1]
解释:
只有红色节点满足条件“所有不包含 1 的子树”。 右图为返回的答案。

示例 2:

img
输入:root = [1,0,1,0,0,0,1]
输出:[1,null,1,null,1]

示例 3:

img
输入:root = [1,1,0,1,1,0,1,0]
输出:[1,1,0,1,1,null,1]

提示:

  • 树中节点的数目在范围 [1, 200]
  • Node.val01

解题思路:深度优先遍历 + 后序遍历

​ 首先要理解这道题的意思,不是说当前节点为零就要剪枝,而是 要看以当前节点为根节点的子树是否都为零,如果是的话才进行剪枝,这个是这道题容易搞错的点!

​ 那么既然我们需要知道当前节点的左右子树情况才进行剪枝,那么势必就要使用 后序遍历 才行,因为后序遍历是最后才处理当前节点,此时左右子树是已经处理完了的,那么我们可以让左右子树直接返回一个指针,以左子树为例,如果左子树整棵子树都是零的话,那么直接返回空指针即可,对于右子树也是如此!

​ 此时后序遍历之后我们也拿到了左右子树返回的两个指针,我们判断一下,如果当前节点为零,并且左右子树都是空指针的话,说明当前节点也是需要被剪枝的,则直接返回一个 nullptr 即可,否则的话返回当前节点给上一层,以此类推!

​ 所以下面分三步来进行讨论:

  1. 函数头的设计

    • 因为我们需要左右子树和当前节点返回一个节点指针,所以返回值就是 TreeNode*。然后因为整个过程其实我们只需要使用到当前节点,所以只需要一个变量来表示当前节点,那么题目给的函数头刚好符合要求,我们就直接拿题目的函数头进行使用,如下所示:

      TreeNode* pruneTree(TreeNode* root);
      
  2. 函数体的内容

    • 首先毋庸置疑就是要进行后序遍历,先递归到左子树,拿到其返回值,判断一下返回值是否为空,为空的话表示左子树就是需要剪枝的,则直接将 root->left 置为空即可,对于右子树来说也是如此。
    • 然后处理完左右子树之后,此时我们 判断一下当前节点是否为零,是的话 再判断一下左右子树是否都为空
      • 如果都为空表示当前节点是需要剪枝的,直接返回一个 nullptr 即可!
      • 如果左右子树至少有一个不为空的话,则说明以当前节点为根节点的子树是不需要剪枝的,则返回当前节点给上一层即可!
  3. 递归函数出口

    • 出口很简单,就是递归到空节点了,直接返回一个 nullptr 即可!
/*** Definition for a binary tree node.* 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) {}* };*/
class Solution {
public:TreeNode* pruneTree(TreeNode* root) {// 递归函数出口if(root == nullptr)return nullptr;// 后序遍历,先处理左右子树,才能知道当前节点是否需要剪枝root->left = pruneTree(root->left);root->right = pruneTree(root->right);// 再处理当前节点,如果当前节点为零并且是叶子节点的话,则返回false即可if(root->val == 0 && root->left == nullptr && root->right == nullptr){delete root;root = nullptr;}return root;}
};

另一种写法

​ 当然,我们也可以把 dfs() 函数单独拿出来,然后用返回值为布尔值的形式进行处理,大体思路都是一样的,只不过函数头不同而导致细节变了一点而已!

/*** Definition for a binary tree node.* 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) {}* };*/
class Solution {
public:TreeNode* pruneTree(TreeNode* root) {if(dfs(root))return root;return nullptr;}bool dfs(TreeNode* root){if(root == nullptr)return false;// 后序遍历,先处理左右子树,才能知道当前节点是否需要剪枝// 如果左右孩子递归处理之后发现可以剪枝的,直接将其置为nullptr即可if(dfs(root->left) == false)root->left = nullptr;if(dfs(root->right) == false)root->right = nullptr;// 再处理当前节点,如果当前节点为零并且是叶子节点的话,则返回false即可if(root->val == 0 && root->left == nullptr && root->right == nullptr){delete root; // 别忘了释放一下节点防止内存泄漏return false;}return true;}
};

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Android SystemUI——自定义状态栏和导航栏(十二)

通过前面的文章内容,我们了解了 Android 系统原生的状态栏 StatusBar 和 车载系统状态栏 CarStatusBar 的启动流程以及视图构建流程,这里我们来简单的看一下自定义状态栏和导航栏视图的实现流程。 一、添加自定义状态栏 修改 CarSystemUI 项目中的 config.xml 配置文件的 co…

CSS实现实现票据效果 mask与切图方式

一、“切图”的局限性 传统的“切图”简单暴力,但往往缺少适应性。 适应性一般有两种,一是尺寸自适应,二是颜色可以自定义。 举个例子,有这样一个优惠券样式 关于这类样式实现技巧,之前在这篇文章中有详细介绍: CSS 实现优惠券的技巧 不过这里略微不一样的地方是,两个…

C语言数组详解:从基础到进阶的全面解析

在C语言中,数组是一种基本的数据结构,用于存储多个相同类型的数据。数组的引入使得C语言能够高效地存储和操作大量数据。在任何一个C语言程序中,数组都发挥着极其重要的作用。无论是在算法实现、数据存储、还是在复杂程序的设计中&#xff0c…

Vue2 项目二次封装Axios

引言 在现代前端开发中,HTTP请求管理是构建健壮应用的核心能力之一。Axios作为目前最流行的HTTP客户端库,其灵活性和可扩展性为开发者提供了强大的基础能力。 1. 为什么要二次封装Axios? 1.1 统一项目管理需求 API路径标准化:…

Jmeter 动态参数压力测试时间段预定接口

🎯 本文档详细介绍了如何使用Apache JMeter进行压力测试,以评估预定接口在高并发场景下的性能表现。通过创建线程组模拟不同数量的用户并发请求,利用CSV文件动态配置时间段ID和用户token,确保了测试数据的真实性和有效性。文档中还…

Unity常用特性(Attribute)用法

一.UnityEngine命名空间 1.[Header(string)] inspector面板上给显示的字段上加一个描述 通常情况下,用于在 Inspector 窗口中创建字段的逻辑分组 public class AttributeTest : MonoBehaviour {[Header("public_field_num")]public int num; }2.[Tool…

vue项目的创建

运行第一个vue-cli应用程序 创建一个基于webpack模板的vue应用程序 vue init webpack 项目名根据自己需求选择 创建好之后如下 运行 cd vue01npm run dev运行之后如下 复制访问地址 : http://localhost:8080 停止服务 两次ctrlC 或者 一次ctrlc然后y idea中使用…

【技术杂谈】Arcgis调用天地图和卫星影像

Arcgis调用天地图和卫星影像 Arcgis调用天地图 1.注册用户官网地址:https://www.tianditu.gov.cn/在官网右上角找到“注册”,输入账号信息完成注册。 2.申请车成为天地图开发者在首页往下滑,找到开发资源下的成为开发者,填写好…

【ROS】RViz2源码分析(四):初始化、启动

【ROS】郭老二博文之:ROS目录 1、简述 RViz2在main函数中,首先注册日志处理函数; 将 RCLCPP_DEBUG 等日志记录函数,通过 rviz_common::set_logging_handlers() 注册到 rviz_common 中。然后,创建界面类 rviz_common::VisualizerApp,并执行初始化 vapp.init(argc, argv)…

【CS61A 2024秋】Python入门课,全过程记录P3(Week5 Sequences开始,更新于2025/1/23)

文章目录 关于基本介绍👋新的问题Week5Mon Sequences阅读材料 关于 个人博客,里面偶尔更新,最近比较忙。发一些总结的帖子和思考。 江湖有缘相见🤝。如果读者想和我交个朋友可以加我好友(见主页or个人博客&#xff0…

android手机应用连接热点后无法进行tcp连接

你在WifiNetworkSpecifer连接回调onavaliable里,再次调用bindProcessToNetwork试试,我这边模拟了一下,是可以建立tcp连接的 你的那个应用我一直没编译成功,你试试吧,应该这样是可以的 另一个同事找到了类似的方法&…

【华为路由的arp配置】

华为路由的arp配置 ARP:IP地址与MAC地址的映射。 R1: g0/0/0:10.1.1.254/24 g0/0/1:10.1.2.254/24 PC1: 10.1.1.1/16 PC2: 10.1.1.2/16 PC3: 10.1.2.3/16 动态ARP 查看PC1的arp表,可以看到,列表为空。 查看R1的arp表 在PC3上ping命令测…

SPDK vhost介绍

目录 1. vhost技术的背景与动机Virtio 介绍virtio-blk数据路径为例 2. vhost技术的核心原理2.1 vhost-kernel2.2 vhost-user举例 2.3 SPDK vhostvhost的优势IO请求处理数据传输控制链路调整 3. SPDK vhost的实现与配置3.1 环境准备3.2 启动SPDK vhost服务3.3 创建虚拟块设备3.4…

GPU 编程系列:内核网格与多维数据处理

GPU 编程系列:内核网格与多维数据处理 大家好,欢迎来到 GPU 编程系列的第三集!在这一集中,我们将深入探讨内核网格的概念,并展示如何利用多维网格来处理复杂的数据结构。 3 内核网格的基础 在上期节目中,…

电容的一些常用数值

如果是滤高频信号的小电容一般采用100nF 如果是滤低频信号的大电容一般采用10uF(10000nF) 比如这个LDO降压 两个一起用滤波效果会更好 如果想要供电引脚悬空,按理不能悬空,所以应该接大电阻接地,一般采用5.1KΩ 比如这个6Pin USB-TypeC的…

Apache Flink 概述学习笔记

一、引言 在大数据处理领域,Apache Flink 是一个极具影响力的开源流批一体化计算框架,它以其独特的架构和强大的功能,为大规模数据处理提供了高效、灵活的解决方案。 二、基本概念 Flink 是什么:Flink 是一个分布式流批处理框架…

Java面向对象专题

面向过程和面向对象的区别 面向过程:当事件比较简单的时候,利用面向过程,注重的是事件的具体的步骤/过程,注重的是过程中的具体的行为,以函数为最小单位,考虑怎么做。 面向对象:注重找“参与者”,将功能封装进对象,强调具备了功能的对象,以类/对象为最小单位,考虑…

业余无线电 对讲机常用频率使用

我自己的总结是,基本可以无忧使用: 144.035-145.800 146.000-148.000 430.000-431.900 432.240-435.000 438.000-439.000 50Mhz一般手台不支持,暂不记录。 以下为附录可以自行阅读,本文内容如有错误请留言指正。 特定波段…

linux 扩容

tmpfs tmpfs 82M 0 82M 0% /run/user/1002 tmpfs tmpfs 82M 0 82M 0% /run/user/0 [输入命令]# fdisk -lu Disk /dev/vda: 40 GiB, 42949672960 bytes, 83886080 sectors Units: sectors of 1 * 512 512 bytes Sector size (logi…

一个基于Python+Appium的手机自动化项目~~

本项目通过PythonAppium实现了抖音手机店铺的自动化询价,可以直接输出excel,并带有详细的LOG输出。 1.excel输出效果: 2. LOG效果: 具体文件内容见GitCode: 项目首页 - douyingoods:一个基于Pythonappium的手机自动化项目,实现了…