1110. 删点成林
给出二叉树的根节点 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. 删点成林相关推荐
- Leetcode 1110.删点成林(Delete Nodes And Return Forest)
Leetcode 1110.删点成林 1 题目描述(Leetcode题目链接) 给出二叉树的根节点 root,树上每个节点都有一个不同的值. 如果节点值在 to_delete 中出现,我们就把该节 ...
- PigyChan_LeetCode 1110. 删点成林
1110. 删点成林 难度中等 给出二叉树的根节点 root,树上每个节点都有一个不同的值. 如果节点值在 to_delete 中出现,我们就把该节点从树上删去,最后得到一个森林(一些不相交的树构成的 ...
- LeetCode 刷题系列 -- 1110. 删点成林
给出二叉树的根节点 root,树上每个节点都有一个不同的值. 如果节点值在 to_delete 中出现,我们就把该节点从树上删去,最后得到一个森林(一些不相交的树构成的集合). 返回森林中的每棵树.你 ...
- leetCode 1110 删点成林(树,后序遍历)
题目链接:点击查看 题目描述: 给定一个整数二叉树和一些整数,求删掉这些整数对应的节点后,剩余的子树. 输入输出: 输入:root = [1,2,3,4,5,6,7], to_delete = [3, ...
- 力扣解法汇总1110. 删点成林
目录链接: 力扣编程题-解法汇总_分享+记录-CSDN博客 GitHub同步刷题项目: GitHub - September26/java-algorithms: 算法题汇总,包含牛客,leetCod ...
- LeetCode 1110. 删点成林(二叉树递归)
1. 题目 给出二叉树的根节点 root,树上每个节点都有一个不同的值. 如果节点值在 to_delete 中出现,我们就把该节点从树上删去,最后得到一个森林(一些不相交的树构成的集合). 返回森林中 ...
- leetcode 1110. 删点成林
题目描述: 给出二叉树的根节点 root,树上每个节点都有一个不同的值. 如果节点值在 to_delete 中出现,我们就把该节点从树上删去,最后得到一个森林(一些不相交的树构成的集合). 返回森林中 ...
- 1110 删点成林
题目描述: 给出二叉树的根节点 root,树上每个节点都有一个不同的值. 如果节点值在 to_delete 中出现,我们就把该节点从树上删去,最后得到一个森林(一些不相交的树构成的集合). 返回森林中 ...
- LeetCode 1110.删点成林
主页有其他数据结构内容(持续更新中) 难度:Medium 代码: /*** Definition for a binary tree node.* struct TreeNode {* int val ...
最新文章
- 4、Power Map—实例:演示报告封面
- java redis快速入门_SpringDataRedis快速入门
- PAT (Advanced Level) 1010 Radix(二分+模拟)
- 如何处理OData错误消息Property XX at offset XX is invalid
- linu - 系统延时与定时任务
- Github无法拉代码
- 安装服务器选择什么系统盘,云服务器ecs选择什么系统盘
- JDK8新特性(八)之Stream流的map()、sorted()、distinct()、match()方法
- leetcode934.ShortestBridge
- 浅谈jQuery Mobile设计思想
- 算法笔记 --- 布隆过滤器
- Boxee智能电视机顶盒在美国CES亮相
- 编程人员应该忙里偷闲
- mysql 命令 kill_当MySQL事务中发生了网络异常
- 【Java练习题】Java 程序的输出 | 第十一套(含解析)
- flash builder java_FlashBuilder4 与 Java 的交互调用(blazeds服务端)
- pycharm安装jpype报错及解决方法
- Windows下Go语言 幽灵蛛的配置
- An exception has been thrown during the rendering of a template
- org.apache.dubbo 2.7.7 服务端处理请求及时间轮(失败重试)
热门文章
- [Bzoj2246]迷宫探险(概率+DP)
- 因果推断(二)-Causal effect 因果效应推理
- 如何寻找论文的创新点
- HDU4507 吉哥系列故事――恨7不成妻 数位DP
- 外商独资企业必须有监事吗
- 字节跳动取消大小周,网友炸了!
- 蓝桥青少竞赛考纲,一文须知【Python 篇】
- python安装详细步骤windows10_【python安装配置教程win10】win10 怎么安装python
- Codeforces Round #529 (Div. 3) F(kul最小生成树)
- 深度学习的Dimension检查