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

文章目录

  • 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;
    }
};

在这里插入图片描述

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

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

相关文章

快速搭建深度学习环境(Linux:miniconda+pytorch+jupyter notebook)

本文基于服务器端环境展开,使用的虚拟终端为Xshell。 miniconda miniconda是Anaconda的轻量版,仅包含Conda和Python,如果只做深度学习,可使用miniconda。 [注]:Anaconda、Conda与Miniconda Conda:创建和管…

01-硬件入门学习/嵌入式教程-CH340C使用教程

前言 CH340C广泛应用于DIY项目和嵌入式开发中,用于USB数据转换和串口通信。本文将详细介绍CH340C的基本功能、引脚接线及使用方法。 CH340C简介 CH340C是一款USB转TTL电平转换器,可以将电脑的USB数据转换成串口数据,方便与单片机&#xff…

PID 控制算法(二):C 语言实现与应用

在本文中,我们将用 C 语言实现一个简单的 PID 控制器,并通过一个示例来演示如何使用 PID 控制算法来调整系统的状态(如温度、速度等)。同时,我们也会解释每个控制参数如何影响系统的表现。 什么是 PID 控制器&#xf…

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

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

vulfocus/fastjson-cnvd_2017_02833复现

漏洞概述 Fastjson 是阿里巴巴开发的一个高性能的 Java 库,用于将 Java 对象转换成 JSON 格式(序列化),以及将 JSON 字符串转换回 Java 对象(反序列化)。 fastjson在解析json的过程中,支持使用type字段来指…

【unity游戏开发之InputSystem——05】PlayerInput组件的介绍(基于unity6开发介绍)

文章目录 前言一、认识PlayerInput1、PlayerInput是什么?2、主要工作原理:3、好处:二、添加PlayerInput组件三、PlayerInput参数相关1、Actions:行为2、Default Scheme:默认启用哪个控制方案3、Auto-Switch:自动切换控制方案4、Default Map:默认行为映射方案5、Ui Input…

Android GLSurfaceView 覆盖其它控件问题 (RK平台)

平台 涉及主控: RK3566 Android: 11/13 问题 在使用GLSurfaceView播放视频的过程中, 增加了一个播放控制面板, 覆盖在视频上方. 默认隐藏setVisibility(View.INVISIBLE);点击屏幕再显示出来. 然而, 在RK3566上这个简单的功能却无法正常工作. 通过缩小视频窗口可以看到, 实际…

【2024 - 年终总结】叶子增长,期待花开

写在前面:本博客仅作记录学习之用,部分图片来自网络,如需引用请注明出处,同时如有侵犯您的权益,请联系删除! 文章目录 前言论博客创作保持2024的记录清单博客科研开源工作生活 总结与展望互动致谢参考 前言…

TangoFlux 本地部署实用教程:开启无限音频创意脑洞

一、介绍 TangoFlux是通过流匹配和 Clap-Ranked 首选项优化,实现超快速、忠实的文本到音频生成的模型。 本模型由 Stability AI 提供支持🚀 TangoFlux 可以在单个 A40 GPU 上在 ~3 秒内生成长达 34.1kHz 的立体声音频。 二、部署 安装方式非常简单 1…

ThreeJS示例教程200+【目录】

Three.js 是一个强大的 JavaScript 库,旨在简化在网页上创建和展示3D图形的过程。它基于 WebGL 技术,但提供了比直接使用 WebGL 更易于使用的API,使得开发者无需深入了解 WebGL 的复杂细节就能创建出高质量的3D内容。 由于目前内容还不多,下面的内容暂时做一个占位。 文章目…

AIGC的企业级解决方案架构及成本效益分析

AIGC的企业级解决方案架构及成本效益分析 一,企业级解决方案架构 AIGC(人工智能生成内容)的企业级解决方案架构是一个多层次、多维度的复杂系统,旨在帮助企业实现智能化转型和业务创新。以下是总结的企业级AIGC解决方案架构的主要组成部分: 1. 技术架构 企业级AIGC解决方…

Blazo-Blazor Web App项目结构

让我们还是从创建项目开始,来一起了解下Blazor Web App的项目情况 创建项目 呈现方式 这里我们可以看到需要选择项目的呈现方式,有以上四种呈现方式 ● WebAssembly ● Server ● Auto(Server and WebAssembly) ● None 纯静态界面静态SSR呈现方式 WebAs…

我谈概率论与数理统计的知识体系

学习概率统计二十多年后,在廖老师的指导下,厘清了各章之间的关系。本来就是一条线两个分支,脉络很清晰。 分支一:从随机现象到样本空间到随机事件再到概率。 从随机事件到随机变量:为了进行定量的数学处理&#xff0…

检查w-form-select 组件是否正确透传了 visible-change 事件

文章目录 1. 下拉框的组件2. 点击下拉框时的逻辑(1)handleVisibleChange 方法(2)getIdentifies 方法 3. 选择关联识别点时的逻辑(1)onRelatedChange 方法 4. 相关 UI 更新5. 总结 在这段代码中,…

校验收货地址是否超出配送范围实战3(day09)

优化用户下单功能,加入校验逻辑,如果用户的收货地址距离商家门店超出配送范围(配送范围为5公里内),则下单失败。 提示: ​ 1. 基于百度地图开放平台实现(https://lbsyun.baidu.com/&#xff09…

Linux应用编程(五)USB应用开发-libusb库

一、基础知识 1. USB接口是什么? USB接口(Universal Serial Bus)是一种通用串行总线,广泛使用的接口标准,主要用于连接计算机与外围设备(如键盘、鼠标、打印机、存储设备等)之间的数据传输和电…

【优选算法】7----三数之和

来了来了,他来了,又是学习算法的一天~ 今天的嘉宾是中等难度的算法题----三数之和! ------------------------------------------begin------------------------------------ 题目解析: 哇趣!又是给了一个数组&#…

深度学习|表示学习|卷积神经网络|参数共享是什么?|07

如是我闻: Parameter Sharing(参数共享)是卷积神经网络(CNN)的一个重要特性,帮助它高效地处理数据。参数共享的本质就是参数“本来也没有变过”。换句话说,在卷积层中,卷积核的参数&…

DeepSeek-R1:性能对标 OpenAI,开源助力 AI 生态发展

DeepSeek-R1:性能对标 OpenAI,开源助力 AI 生态发展 在人工智能领域,大模型的竞争一直备受关注。最近,DeepSeek 团队发布了 DeepSeek-R1 模型,并开源了模型权重,这一举动无疑为 AI 领域带来了新的活力。今…

递归的本质

字节面试题叠罗汉,很遗憾没想出来,看了答案挺巧妙的,但是居然是个案例题。。。 复习一下递归的本质 正面解决问题 利用子问题来解决 可以通过规约推导的,基本可以用递归解决! 在写这道算法题时,我想规…