给出二叉树的根节点 root,树上每个节点都有一个不同的值。

如果节点值在 to_delete 中出现,我们就把该节点从树上删去,最后得到一个森林(一些不相交的树构成的集合)。

返回森林中的每棵树。你可以按任意顺序组织答案。

示例 1:

输入:root = [1,2,3,4,5,6,7], to_delete = [3,5]
输出:[[1,2,null,4],[6],[7]]
示例 2:

输入:root = [1,2,4,null,3], to_delete = [3]
输出:[[1,2,4]]

提示:

树中的节点数最大为 1000。
每个节点都有一个介于 1 到 1000 之间的值,且各不相同。
to_delete.length <= 1000
to_delete 包含一些从 1 到 1000、各不相同的值。

写在前面:致力于写出最好理解的题解。

解题思路
对于被删除节点类型,可以分情况进行分析:

叶子节点:直接删掉
非叶子节点:如果左右子树不为空,就加入结果集。
为了得到这个结果集,每个结点需要做什么?

每个结点要知道自己是否需要被删掉,如果需要被删掉,那么就根据情况执行上一步的分析。
在什么时候做这些事?

如果一个结点需要被删掉,那么应该将它与父节点的关系进行移除,所以我们需要:判断完结点是否需要被删除后,告知他的父节点进行相应的指针操作。

那么不妨采用后序遍历的方式。

定义一个删除函数

/**
* 定义:
* 删除以node为根的树中,结点值出现在delete中的结点,更新结果集list,并返回删除后新的根节点。
*/
TreeNode del(TreeNode node, Set<Integer> delete, List<TreeNode> list);
把上述思考过程直接翻译成代码:

TreeNode del(TreeNode node, Set<Integer> delete, List<TreeNode> list){
        if(node == null) return null; // base case
        // 先递归左右子树
        node.left = del(node.left, delete, list);
        node.right = del(node.right, delete,list);
        
        if(delete.contains(node.val)){ // 当node结点需要被删掉时,更新结果集
            if(node.left != null){ 
                list.add(node.left);
            }
            if(node.right != null){ 
                list.add(node.right);
            }
            // 如果结果集里已经包含node为根的树,就把结果集里的node删掉
            if(list.contains(node)){
                list.remove(node);
            }
            // 将node置为null,等于删除掉了
            return null;
        }
        return node;
    }
那么在执行del()方法前,我们需要做什么准备工作呢?

构建一个 delete 集合
将最初的根节点加入结果集,这么做的原因是:如果没有要删除的结点,那么del方法是不会将根节点加入结果集的,所以我们需要提前加入,并且在del方法中会维护指针索引,所以结果集中的树结构会保证正确。

    public List<TreeNode> delNodes(TreeNode root, int[] to_delete) {List<TreeNode> list = new LinkedList<>();Set<Integer> set = new HashSet<>();// 将delete转换为set方便进行判断for(int num : to_delete){set.add(num);}// 如果什么都不需要删掉,那么结果集中应该包含原本这棵树list.add(root);del(root, set, list);return list;}TreeNode del(TreeNode node, Set<Integer> delete, List<TreeNode> list){if(node == null) return null;node.left = del(node.left, delete, list);node.right = del(node.right, delete,list);if(delete.contains(node.val)){if(node.left != null){list.add(node.left);}if(node.right != null){list.add(node.right);}if(list.contains(node)){list.remove(node);}return null;}return node;}

解题思路

findNodes函数解决了连环删除的情况,就是父子节点都要被删除

class Solution {Set<Integer> to_delete_set;List<TreeNode> res; public List<TreeNode> delNodes(TreeNode root, int[] to_delete) {to_delete_set = new HashSet<>();for(int i : to_delete){to_delete_set.add(i);}res = new LinkedList<>();findNodes(root);return res;}public void findNodes(TreeNode root){// 如果当前节点是删除的,那跳过(由于母节点在调用时要么为空要么也为被删除,因此不需要断开树),如果不删除,那么首先加入res,然后BFS寻找需要删除的if(root == null){return;}if(to_delete_set.contains(root.val)){// 解决了连环删除的情况findNodes(root.left);findNodes(root.right);return;}res.add(root);List<TreeNode> layer = new LinkedList<>();layer.add(root);while(!layer.isEmpty()){List<TreeNode> nextLayer = new LinkedList<>();for(TreeNode node : layer){if(node == null){continue;}if(node.left != null){// 因为要断开树,因此验证的是子节点if(to_delete_set.contains(node.left.val)){findNodes(node.left.left);findNodes(node.left.right);node.left = null;}else{nextLayer.add(node.left);}}if(node.right != null){if(to_delete_set.contains(node.right.val)){findNodes(node.right.left);findNodes(node.right.right);node.right = null;}else{nextLayer.add(node.right);}}}layer = nextLayer;}}
}

1110. 删点成林相关推荐

  1. Leetcode 1110.删点成林(Delete Nodes And Return Forest)

    Leetcode 1110.删点成林 1 题目描述(Leetcode题目链接)   给出二叉树的根节点 root,树上每个节点都有一个不同的值. 如果节点值在 to_delete 中出现,我们就把该节 ...

  2. PigyChan_LeetCode 1110. 删点成林

    1110. 删点成林 难度中等 给出二叉树的根节点 root,树上每个节点都有一个不同的值. 如果节点值在 to_delete 中出现,我们就把该节点从树上删去,最后得到一个森林(一些不相交的树构成的 ...

  3. LeetCode 刷题系列 -- 1110. 删点成林

    给出二叉树的根节点 root,树上每个节点都有一个不同的值. 如果节点值在 to_delete 中出现,我们就把该节点从树上删去,最后得到一个森林(一些不相交的树构成的集合). 返回森林中的每棵树.你 ...

  4. leetCode 1110 删点成林(树,后序遍历)

    题目链接:点击查看 题目描述: 给定一个整数二叉树和一些整数,求删掉这些整数对应的节点后,剩余的子树. 输入输出: 输入:root = [1,2,3,4,5,6,7], to_delete = [3, ...

  5. ​力扣解法汇总1110. 删点成林

    目录链接: 力扣编程题-解法汇总_分享+记录-CSDN博客 GitHub同步刷题项目: GitHub - September26/java-algorithms: 算法题汇总,包含牛客,leetCod ...

  6. LeetCode 1110. 删点成林(二叉树递归)

    1. 题目 给出二叉树的根节点 root,树上每个节点都有一个不同的值. 如果节点值在 to_delete 中出现,我们就把该节点从树上删去,最后得到一个森林(一些不相交的树构成的集合). 返回森林中 ...

  7. leetcode 1110. 删点成林

    题目描述: 给出二叉树的根节点 root,树上每个节点都有一个不同的值. 如果节点值在 to_delete 中出现,我们就把该节点从树上删去,最后得到一个森林(一些不相交的树构成的集合). 返回森林中 ...

  8. 1110 删点成林

    题目描述: 给出二叉树的根节点 root,树上每个节点都有一个不同的值. 如果节点值在 to_delete 中出现,我们就把该节点从树上删去,最后得到一个森林(一些不相交的树构成的集合). 返回森林中 ...

  9. LeetCode 1110.删点成林

    主页有其他数据结构内容(持续更新中) 难度:Medium 代码: /*** Definition for a binary tree node.* struct TreeNode {* int val ...

最新文章

  1. 4、Power Map—实例:演示报告封面
  2. java redis快速入门_SpringDataRedis快速入门
  3. PAT (Advanced Level) 1010 Radix(二分+模拟)
  4. 如何处理OData错误消息Property XX at offset XX is invalid
  5. linu - 系统延时与定时任务
  6. Github无法拉代码
  7. 安装服务器选择什么系统盘,云服务器ecs选择什么系统盘
  8. JDK8新特性(八)之Stream流的map()、sorted()、distinct()、match()方法
  9. leetcode934.ShortestBridge
  10. 浅谈jQuery Mobile设计思想
  11. 算法笔记 --- 布隆过滤器
  12. Boxee智能电视机顶盒在美国CES亮相
  13. 编程人员应该忙里偷闲
  14. mysql 命令 kill_当MySQL事务中发生了网络异常
  15. 【Java练习题】Java 程序的输出 | 第十一套(含解析)
  16. flash builder java_FlashBuilder4 与 Java 的交互调用(blazeds服务端)
  17. pycharm安装jpype报错及解决方法
  18. Windows下Go语言 幽灵蛛的配置
  19. An exception has been thrown during the rendering of a template
  20. org.apache.dubbo 2.7.7 服务端处理请求及时间轮(失败重试)

热门文章

  1. [Bzoj2246]迷宫探险(概率+DP)
  2. 因果推断(二)-Causal effect 因果效应推理
  3. 如何寻找论文的创新点
  4. HDU4507 吉哥系列故事――恨7不成妻 数位DP
  5. 外商独资企业必须有监事吗
  6. 字节跳动取消大小周,网友炸了!
  7. 蓝桥青少竞赛考纲,一文须知【Python 篇】
  8. python安装详细步骤windows10_【python安装配置教程win10】win10 怎么安装python
  9. Codeforces Round #529 (Div. 3) F(kul最小生成树)
  10. 深度学习的Dimension检查