代码随想录算法训练营第22天|● 235. 二叉搜索树的最近公共祖先 ● 701.二叉搜索树中的插入操作 ● 450.删除二叉搜索树中的节点

二叉搜索树的最近公共祖先

题目链接

https://leetcode.cn/problems/lowest-common-ancestor-of-a-binary-search-tree/description/

思路

因为是有序树,所以 如果 中间节点是 q 和 p 的公共祖先,那么 中节点的数组 一定是在 [p, q]区间的。即 中节点 > p && 中节点 < q 或者 中节点 > q && 中节点 < p。所以子需要从上到下遍历第一次遇见符合条件的就是最小的公共祖先

代码:

java 复制代码
/**
 * Definition for a binary tree node.
 * public class TreeNode {
 *     int val;
 *     TreeNode left;
 *     TreeNode right;
 *     TreeNode(int x) { val = x; }
 * }
 */

class Solution {
    public TreeNode lowestCommonAncestor(TreeNode root, TreeNode p, TreeNode q) {
           if(root==null){
             return null;
         }
        if(root.val>p.val&&root.val>q.val){//root.val比p,q的值都大,说明最小祖先在左子树
            TreeNode left=lowestCommonAncestor(root.left,p,q);
            if(left!=null){
                return left;
            }
        }
        if(root.val<p.val&&root.val<q.val){//root.val比p,q的值都小,说明最小祖先在右子树
            TreeNode right=lowestCommonAncestor(root.right,p,q);
            if(right!=null){
                return right;
            }
        }
        return root;
    }
}

二叉搜索树中的插入操作

题目链接

https://leetcode.cn/problems/insert-into-a-binary-search-tree/

思路:

基本情况:如果传入的树根节点root为null,表示树为空。此时,函数创建一个新的TreeNode对象,用val初始化,并返回这个新节点作为新的树根。

递归情况:如果树不为空,函数会根据val和当前节点root的值进行操作:

如果val小于root.val,说明val应该被插入到左子树中。因此,函数会递归调用自身,将val插入到root.left(左子节点),并将更新后的左子树赋值回root.left。

如果val大于root.val,说明val应该被插入到右子树中。同理,函数会递归调用自身,将val插入到root.right(右子节点),并将更新后的右子树赋值回root.right。

返回结果:无论哪种情况,函数最后都会返回当前节点root。这使得递归调用能够逐级更新原树结构,直到插入操作完成并返回最初的树根节点。

代码:

java 复制代码
/**
 * Definition for a binary tree node.
 * public class TreeNode {
 *     int val;
 *     TreeNode left;
 *     TreeNode right;
 *     TreeNode() {}
 *     TreeNode(int val) { this.val = val; }
 *     TreeNode(int val, TreeNode left, TreeNode right) {
 *         this.val = val;
 *         this.left = left;
 *         this.right = right;
 *     }
 * }
 */
class Solution {
    public TreeNode insertIntoBST(TreeNode root, int val) {
             if(root==null){
            TreeNode node=new TreeNode(val);
            return node;
        }
        if(root.val>val){
            root.left=insertIntoBST(root.left,val);
        }
        if(root.val<val){
            root.right=insertIntoBST(root.right,val);
        }
        return root;
    }
}

删除二叉搜索树中的节点

题目链接

https://leetcode.cn/problems/delete-node-in-a-bst/

思路:

首先判断根节点是否为空,若为空则返回null。然后判断根节点的值是否等于要删除的值,若相等则分四种情况处理:若该节点没有左右子节点,则直接返回null;若只有左子节点,则返回左子节点;若只有右子节点,则返回右子节点;若既有左子节点又有右子节点,则找到右子树中最小的节点,将其值替换要删除的节点的值,并删除右子树中最小的节点。接着判断要删除的值是否小于根节点的值,若是则递归地在左子树中删除该值;若大于根节点的值,则递归地在右子树中删除该值。最后返回修改后的根节点。

代码:

java 复制代码
/**
 * Definition for a binary tree node.
 * public class TreeNode {
 * int val;
 * TreeNode left;
 * TreeNode right;
 * TreeNode() {}
 * TreeNode(int val) { this.val = val; }
 * TreeNode(int val, TreeNode left, TreeNode right) {
 * this.val = val;
 * this.left = left;
 * this.right = right;
 * }
 * }
 */
class Solution {
    public TreeNode deleteNode(TreeNode root, int key) {
     // 如果根节点为空,直接返回null
        if (root == null) {
            return null;
        }
        // 如果找到需要删除的节点
        if (root.val == key) {
            // 如果该节点没有子节点,直接删除(返回null)
            if(root.left==null&&root.right==null){
                return null;
            }
            // 如果该节点只有左子节点,用左子节点替换该节点后返回
            else if(root.left!=null&&root.right==null){
                return root.left;
            }
            // 如果该节点只有右子节点,用右子节点替换该节点后返回
            else if(root.left==null&&root.right!=null){
                return root.right;
            }
            // 如果该节点既有左子节点又有右子节点,找到右子树中最小的节点,将其值替换该节点的值,然后递归删除右子树中的该最小节点
            else if(root.left!=null&&root.right!=null){
                TreeNode temp=root.right;
                while (temp.left!=null){
                    temp=temp.left;
                }
                temp.left=root.left;
                return root.right;
            }
        }
        // 如果当前节点的值大于目标值,递归删除左子树
        if(root.val>key){
            root.left=deleteNode(root.left,key);
        }
        // 如果当前节点的值小于目标值,递归删除右子树
        if(root.val<key){
            root.right=deleteNode(root.right,key);
        }
        // 返回修改后的树的根节点
        return root;
    }
}
相关推荐
知己如祭3 分钟前
图论基础(DFS、BFS、拓扑排序)
算法
mit6.82412 分钟前
[Cyclone] 哈希算法 | SIMD优化哈希计算 | 大数运算 (Int类)
算法·哈希算法
c++bug15 分钟前
动态规划VS记忆化搜索(2)
算法·动态规划
哪 吒17 分钟前
2025B卷 - 华为OD机试七日集训第5期 - 按算法分类,由易到难,循序渐进,玩转OD(Python/JS/C/C++)
python·算法·华为od·华为od机试·2025b卷
军训猫猫头1 小时前
1.如何对多个控件进行高效的绑定 C#例子 WPF例子
开发语言·算法·c#·.net
success1 小时前
【爆刷力扣-数组】二分查找 及 衍生题型
算法
CodeWithMe2 小时前
【Note】《深入理解Linux内核》 Chapter 15 :深入理解 Linux 页缓存
linux·spring·缓存
0wioiw02 小时前
Ubuntu基础(监控重启和查找程序)
linux·服务器·ubuntu
Tipriest_2 小时前
Ubuntu常用的软件格式deb, rpm, dmg, AppImage等打包及使用方法
linux·运维·ubuntu
Orlando cron2 小时前
数据结构入门:链表
数据结构·算法·链表