《剑指 Offer》专项突破版 - 面试题 47 : 二叉树剪枝(C++ 实现)

题目链接LCR 047. 二叉树剪枝 - 力扣(LeetCode)

题目

一棵二叉树的所有节点的值要么是 0 要么是 1,请剪除该二叉树中所有节点的值全都是 0 的子树。例如,在剪除下图 (a) 中二叉树中所有节点值都为 0 的子树之后的结果如下图 (b) 所示。

分析

首先分析哪些子树会被剪除,哪些子树不能被剪除。在上图 (a) 的二叉树中,以第 2 层第 1 个节点为根节点的子树的 3 个节点的值都是 0,因此整棵二叉树都被剪除。以第 2 层第 2 个节点为根节点的子树中有一个节点的值是 1(第 3 层第 4 个节点),因此这棵子树不能删除,但是它的左子树(根节点为第 3 层第 3 个节点)只有一个节点并且值为 0,因此它的左子树可以被剪除。

下面总结什么样的节点可以被删除。首先,这个节点的值应该是 0。其次,如果它有子树,那么它的子树的所有节点的值都为 0。也就是说,如果一个节点可以被删除,那么它的子树的所有节点都可以被删除。例如,在上图 (a) 的二叉树中,第 2 层第 1 个节点可以被删除,它的子树中的所有节点(第 3 层第 1 个节点和第 2 个节点)也可以被删除。

由此发现,后序遍历 最适合用来解决这个问题。如果用后序遍历的顺序遍历到某个节点,那么它的左右子树的节点一定已经遍历过了。每遍历到一个节点,就要确定它是否有左右子树,如果左右子树都是空的,并且节点的值是 0,那么就可以删除这个节点。

代码实现

cpp 复制代码
class Solution {
public:
    TreeNode* pruneTree(TreeNode* root) {
        if (!root)
            return nullptr;
        
        root->left = pruneTree(root->left);
        root->right = pruneTree(root->right);
        if (!root->left && !root->right && root->val == 0)
            return nullptr;
        
        return root;
    }
};
相关推荐
wkj0013 分钟前
php操作redis
开发语言·redis·php
极客代码8 分钟前
【Python TensorFlow】进阶指南(续篇三)
开发语言·人工智能·python·深度学习·tensorflow
土豆湿14 分钟前
拥抱极简主义前端开发:NoCss.js 引领无 CSS 编程潮流
开发语言·javascript·css
小陈phd16 分钟前
Vscode LinuxC++环境配置
linux·c++·vscode
界面开发小八哥21 分钟前
更高效的Java 23开发,IntelliJ IDEA助力全面升级
java·开发语言·ide·intellij-idea·开发工具
火山口车神丶30 分钟前
某车企ASW面试笔试题
c++·matlab
jiao_mrswang41 分钟前
leetcode-18-四数之和
算法·leetcode·职场和发展
qystca1 小时前
洛谷 B3637 最长上升子序列 C语言 记忆化搜索->‘正序‘dp
c语言·开发语言·算法
薯条不要番茄酱1 小时前
数据结构-8.Java. 七大排序算法(中篇)
java·开发语言·数据结构·后端·算法·排序算法·intellij-idea
今天吃饺子1 小时前
2024年SCI一区最新改进优化算法——四参数自适应生长优化器,MATLAB代码免费获取...
开发语言·算法·matlab