力扣日记1.11-【二叉树篇】450. 删除二叉搜索树中的节点

力扣日记:【二叉树篇】450. 删除二叉搜索树中的节点

日期:2024.1.11
参考:代码随想录、力扣

450. 删除二叉搜索树中的节点

题目描述

难度:中等

给定一个二叉搜索树的根节点 root 和一个值 key,删除二叉搜索树中的 key 对应的节点,并保证二叉搜索树的性质不变。返回二叉搜索树(有可能被更新)的根节点的引用。

一般来说,删除节点可分为两个步骤:

  1. 首先找到需要删除的节点;
  2. 如果找到了,删除它。

示例 1:
力扣日记1.11-【二叉树篇】450. 删除二叉搜索树中的节点_第1张图片

输入:root = [5,3,6,2,4,null,7], key = 3
输出:[5,4,6,2,null,null,7]
解释:给定需要删除的节点值是 3,所以我们首先找到 3 这个节点,然后删除它。
一个正确的答案是 [5,4,6,2,null,null,7], 如上图所示。
另一个正确答案是 [5,2,6,null,4,null,7]。
力扣日记1.11-【二叉树篇】450. 删除二叉搜索树中的节点_第2张图片

示例 2:

输入: root = [5,3,6,2,4,null,7], key = 0
输出: [5,3,6,2,4,null,7]
解释: 二叉树不包含值为 0 的节点

示例 3:

输入: root = [], key = 0
输出: []

提示:

  • 节点数的范围 [0, 10^4].
  • -10^5 <= Node.val <= 10^5
  • 节点值唯一
  • root 是合法的二叉搜索树
  • -10^5 <= key <= 10^5

进阶: 要求算法时间复杂度为 O(h),h 为树的高度。

题解

/**
 * 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 {
#define SOLUTION 1
public:
#if SOLUTION == 1
    TreeNode* deleteNode(TreeNode* root, int key) {
        // 空树直接返回(或者找到底也没找到key)
        if (root == nullptr) return root;
        // 找到了,开始删除
        if (root->val == key) {
            // 如果左节点为空,直接返回右节点
            if (root->left == nullptr)  return root->right; // 右节点可能为空可能不为空
            // 如果右节点为空,直接返回左节点
            else if (root->right == nullptr)    return root->left;  // 左节点不为空
            // 这里则是左右节点都不为空
            else {
                // 如果root的左节点没有右子节点,可以把右子树直接作为左节点的右子树
                if (root->left->right == nullptr) {
                    root->left->right = root->right;
                    return root->left;  // 返回root的左节点作为新的根节点
                } 
                // 如果root左节点存在右子节点,则看root的右节点的左子节点
                else if (root->right->left == nullptr) {
                    root->right->left = root->left;
                    return root->right; // 返回root的右节点作为新的根节点
                }
                else {
                    // 这里则是root的左节点有右子节点,且右节点有左子节点
                    // 把root右子树接到root左子树中
                    // 循环找到root的左节点的右子节点为空(最右节点的值是最大的)
                    TreeNode* cur = root->left;
                    while (cur->right != nullptr) {
                        cur = cur->right;
                    }
                    // 此时cur->right为空
                    cur->right = root->right;
                    return root->left; // 把root的左节点返回
                }
            }
        }
        // 没找到,看大小,继续递归
        if (key > root->val) {  // 往右找
            root->right = deleteNode(root->right, key); // 找到会返回新的右子树根节点
        } else {  // 另一种情况就是往左找
            root->left = deleteNode(root->left, key);   // 
        }
        return root;
    }
#endif
};

复杂度

时间复杂度:
空间复杂度:
力扣日记1.11-【二叉树篇】450. 删除二叉搜索树中的节点_第3张图片

思路总结

  • 能解出题来好开心(/(ㄒoㄒ)/~~,虽然过程和代码写的很不简洁,但太难得了(悲
  • 注释的过程即为解题思路过程,可以多画点图模拟一下
  • 关于找到key后删除当前root节点的思路,分几种情况:
      1. 如果root左节点为空,直接返回右节点
      1. 如果root右节点为空,直接返回左节点
      1. 如果左右节点都不为空
      • 1)首先:考虑 如果root的左节点没有右子节点,可以把右子树直接作为左节点的右子树
      • 2)如果root左节点存在右子节点,则看root的右节点的左子节点,思路同理
      • 3)如果都不满足,即root的左节点有右子节点,且右节点有左子节点
        • 则考虑把root右子树接到root左子树中
        • 通过 不断迭代 root的左子树中的右子节点,直到找到 右子节点为空(因为最最右的右子节点的值是左子树中最大的)
        • 找到后,把root的右子节点接到该空节点处,并返回root的左节点
  • 关于删除后返回节点或者递归接收节点的思路:与插入是类似的,都是假定递归函数返回操作(如这里的删除以及上一题的插入)后的新子树根节点,作为当前root节点的新子节点。
  • 如果还未找到key,则可根据二叉搜索树的性质,根据key的大小往左或往右递归寻找,直到递归到空节点则直接返回nullptr。
  • 改进:实际上,对于删除节点时root左右节点都不为空的情况(即第3点),可以都作为第三种情况来考虑(即第3)点),即直接考虑将root右子树接到root左子树中(左子树的右子节点为空则直接接到该空节点处即可,否则就进行迭代找到空右子节点)

你可能感兴趣的:(leetcode,算法,职场和发展)