我是靠谱客的博主 平淡火,最近开发中收集的这篇文章主要介绍669. 修剪二叉搜索树,觉得挺不错的,现在分享给大家,希望可以做个参考。

概述

669. 修剪二叉搜索树

题目链接:https://leetcode-cn.com/problems/trim-a-binary-search-tree/

给你二叉搜索树的根节点 root ,同时给定最小边界low 和最大边界 high。通过修剪二叉搜索树,使得所有节点的值在[low, high]中。修剪树 不应该 改变保留在树中的元素的相对结构 (即,如果没有被移除,原有的父代子代关系都应当保留)。 可以证明,存在 唯一的答案 。

所以结果应当返回修剪好的二叉搜索树的新的根节点。注意,根节点可能会根据给定的边界发生改变。

 

 

思路:

递归法
直接想法就是:递归处理,然后遇到 root.val < low || root.val > high 的时候直接return NULL,一波修改,干净利落。

不难写出如下代码:

``` Java
class Solution {
public TreeNode trimBST(TreeNode root, int low, int high) {
     if (root == null || root.val < low || root.val > high){  return null;

}
     root.left = trimBST(root.left, low, high);
     root.right = trimBST(root.right, low, high);
     return root;
  }
};
```

然而[1, 3]区间在二叉搜索树的中可不是单纯的节点3和左孩子节点0就决定的,还要考虑节点0的右子树

我们在重新关注一下第二个示例,如图:

 

所以以上的代码是不可行的!


从图中可以看出需要重构二叉树,想想是不是本题就有点复杂了。

其实不用重构那么复杂。

在上图中我们发现节点0并不符合区间要求,那么将节点0的右孩子 节点2 直接赋给 节点3的左孩子就可以了(就是把节点0从二叉树中移除),如图:
从图中可以看出需要重构二叉树,想想是不是本题就有点复杂了。

其实不用重构那么复杂。

在上图中我们发现节点0并不符合区间要求,那么将节点0的右孩子 节点2 直接赋给 节点3的左孩子就可以了(就是把节点0从二叉树中移除),如图:

 

理解了最关键部分了我们在递归三部曲:

确定递归函数的参数以及返回值

这里我们为什么需要返回值呢?

因为是要遍历整棵树,做修改,其实不需要返回值也可以,我们也可以完成修剪(其实就是从二叉树中移除节点)的操作。

但是有返回值,更方便,可以通过递归函数的返回值来移除节点。

代码如下:

```
TreeNode trimBST(TreeNode root, int low, int high)
```

确定终止条件

修剪的操作并不是在终止条件上进行的,所以就是遇到空节点返回就可以了。

```
if (root == null ) return null;
```

确定单层递归的逻辑

如果root(当前节点)的元素小于low的数值,那么应该递归右子树,并返回右子树符合条件的头结点。

代码如下:

```
if (root.val < low) {
    TreeNode right = trimBST(root.right, low, high); // 寻找符合区间[low, high]的节点
    return right;
}
```

如果root(当前节点)的元素大于high的,那么应该递归左子树,并返回左子树符合条件的头结点。

代码如下:

```
if (root.val > high) {
    TreeNode left = trimBST(root.left, low, high); // 寻找符合区间[low, high]的节点
    return left;
}
```

接下来要将下一层处理完左子树的结果赋给root.left,处理完右子树的结果赋给root.right。

最后返回root节点,代码如下:

```
root.left = trimBST(root.left, low, high); // root->left接入符合条件的左孩子
root.right = trimBST(root.right, low, high); // root->right接入符合条件的右孩子
return root;
```

此时大家是不是还没发现这多余的节点究竟是如何从二叉树中移除的呢?

在回顾一下上面的代码,针对下图中二叉树的情况:

 

如下代码相当于把节点0的右孩子(节点2)返回给上一层,
```
if (root.val < low) {
    TreeNode right = trimBST(root.right, low, high); // 寻找符合区间[low, high]的节点
    return right;
}
```

然后如下代码相当于用节点3的左孩子 把下一层返回的 节点0的右孩子(节点2) 接住。

```
root.left = trimBST(root.left, low, high);
```

此时节点3的右孩子就变成了节点2,将节点0从二叉树中移除了。

最后整体代码如下:

```Java
class Solution {
    public TreeNode trimBST(TreeNode root, int low, int high) {
        if (root == null) {
            return null;
        }
        if (root.val < low) {
            return trimBST(root.right, low, high);
        }
        if (root.val > high) {
            return trimBST(root.left, low, high);
        }
        // root在[low,high]范围内
        root.left = trimBST(root.left, low, high);
        root.right = trimBST(root.right, low, high);
        return root;
    }
}

```

最后

以上就是平淡火为你收集整理的669. 修剪二叉搜索树的全部内容,希望文章能够帮你解决669. 修剪二叉搜索树所遇到的程序开发问题。

如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(39)

评论列表共有 0 条评论

立即
投稿
返回
顶部