二叉搜索树(二叉排序树)—Java

🔎概念

二叉搜索树又称二叉排序树
可以是一棵空树
也可以不是一棵空树(doge)

在这里插入图片描述
上图所示就是一棵二叉搜索树
根节点root值为7,root的左子树的值全部比root的值小,root的右子树的值全部比root的值大
root.left–>root的左子树的根节点4,其左侧节点的值1比4小,右侧节点的值6比4大(但它们均小于7)
root.right–>root的右子树的根节点11,其左侧节点的值9比11小,右侧节点的值13比11大(但它们均大于7)


🔎模拟实现二叉树

🌻find–>查找二叉树中的元素

public TreeNode find(int val) {
        TreeNode cur = root;
        while(cur != null) {
            if(cur.val < val) {
                cur = cur.right;
            }else if(cur.val > val) {
                cur = cur.left;
            }else {//cur.val == val-->find!
                return cur;
            }
        }
        //未找到
        return null;
}

查找二叉搜索树中指定的val值
如果cur节点的值小于要查找的元素的值,cur = cur.right–>去树的右侧查找
在这里插入图片描述
如果cur节点的值大于要查找的元素的值,cur = cur.left–>去树的左侧查找
在这里插入图片描述


🌻insert–>在二叉树中插入元素

public void insert(int val) {
        if(root == null) {
            root = new TreeNode(val);
            return;
        }
        TreeNode pre = null;//叶子节点
        TreeNode cur = root;

        while(cur != null) {
            pre = cur;
            if(cur.val < val) {
                cur = cur.right;
            }else if(cur.val > val){
                cur = cur.left;
            }else {//二叉搜索树中不要插入相同的数据(无意义)
                return;
            }
        }
        if(val < pre.val) {
            pre.left = new TreeNode(val);
        }else {
            pre.right = new TreeNode(val);
        }
}

如果该搜索树是一棵空树,则插入的元素就是这棵树的root(根节点)
如果不是,在叶子节点的左/右侧插入对应的元素
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述


🌻inorder–>中序遍历二叉树

public void inOrder(TreeNode root) {
        if(root == null) return;
        inOrder(root.left);
        System.out.print(root.val + " ");
        inOrder(root.right);
}

🌻remove–>删除二叉树中的元素

cur–>要删除的节点 / pre–>要删除节点的父节点 / root–>根节点

  • 二叉搜索树的删除
  • cur.left == null–>要删除节点的左侧为空
    1)cur是根节点,root = cur.right
    2)cur不是根节点,cur是父节点的左,pre.left = cur.right
    3)cur不是根节点,cur是父节点的右,pre.right = cur.right
  • cur.right == null–>要删除节点的右侧为空
    1)cur是根节点,root = cur.left
    2)cur不是根节点,cur是父节点的左,pre.left = cur.left
    3)cur不是根节点,cur是父节点的右,pre.right = cur.left
  • cur.left != null && cur.right != null–>要删除节点的左右均不为空
    1)找cur左子树的右叶子节点–>左子树的最大值–>最大值的右节点一定为null
    2)找cur右子树的左叶子节点–>右子树的最小值–>最小值的左节点一定为null
    3)不可以是左子树的左叶子节点(左子树最小值) or 右子树的右叶子节点(右子树最大值)

🌼cur.left == null–>要删除节点的左侧为空

cur是根节点,root = cur.right
在这里插入图片描述
在这里插入图片描述
cur不是根节点,cur是pre节点的左,pre.left = cur.righ
在这里插入图片描述
在这里插入图片描述
cur不是根节点,cur是pre节点的右,pre.right = cur.right
在这里插入图片描述
在这里插入图片描述


🌼cur.right == null–>要删除节点的右侧为空

cur是根节点,root = cur.left
在这里插入图片描述
在这里插入图片描述
cur不是根节点,cur是pre节点的左,pre.left = cur.left
在这里插入图片描述
在这里插入图片描述
cur不是根节点,cur是pre节点的右,pre.right = cur.left
在这里插入图片描述
在这里插入图片描述


🌼cur.left != null && cur.right != null–>要删除节点的左右均不为空

找cur左子树的右叶子节点–>左子树的最大值–>最大值的右节点一定为null
在这里插入图片描述
找cur右子树的左叶子节点–>右子树的最小值–>最小值的左节点一定为null
在这里插入图片描述
不可以是左子树的左叶子节点(左子树最小值) or 右子树的右叶子节点(右子树最大值)
在这里插入图片描述


🌼代码演示

在这里插入图片描述

以cur的右子树的最小值举例

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

以cur的左子树的最大值举例
在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述


🌼完整remove代码

public void remove(int val) {
        TreeNode cur = root;
        TreeNode pre = null;
        while(cur != null) {
            if(cur.val == val) {//找到了值为val的节点
                removeNode(pre,cur);
                return;
            }
            else if(cur.val < val) {
                pre = cur;
                cur = cur.right;
            }else {//cur.val > val
                pre = cur;
                cur = cur.left;
            }
        }
}
//删除值为val的cur节点
private void removeNode(TreeNode pre,TreeNode cur) {
        if(cur.left == null) {
            if(cur == root) {
                root = cur.right;
            }else if(pre.left == cur){//cur在父节点左侧
                pre.left = cur.right;
            }else {//pre.right = cur --> //cur在父节点右侧
                pre.right = cur.right;
            }
        }else if(cur.right == null) {
            if(cur == root) {
                root = cur.left;
            }else if(pre.left == cur) {//cur在父节点左侧
                pre.left = cur.left;
            }else {//pre.right = cur --> //cur在父节点右侧
                pre.right = cur.left;
            }
        }else {//cur.left != null && cur.right != null
            TreeNode targetPre = cur;
            TreeNode target = cur.right;
            while(target.left != null) {//以cur的右子树的最小值举例-->右子树的左叶子节点
                //也可以是cur的左子树的最大值-->左子树的右叶子节点
                targetPre = target;
                target = target.left;
            }
            cur.val = target.val;
            //此时target的左侧一定为null-->target.left == null
            targetPre.left = target.right;//因为是寻找cur右子树的最小值
            //一定是target == targetPre.left
            //target == targetPre.right-->这种情况只会出现在寻找cur左子树的最大值
        }
}

🌻完整模拟代码

public class TestBinarySearch {
    static class TreeNode {
        public int val;
        public TreeNode left;
        public TreeNode right;

        public TreeNode(int val) {
            this.val = val;
        }
    }
    public TreeNode root = null;//根节点
    //查找二叉搜索树中指定的val值
    public TreeNode find(int val) {
        TreeNode cur = root;
        while(cur != null) {
            if(cur.val < val) {
                cur = cur.right;
            }else if(cur.val > val) {
                cur = cur.left;
            }else {//cur.val == val-->find!
                return cur;
            }
        }
        //未找到
        return null;
    }
    //插入元素
    public void insert(int val) {
        if(root == null) {
            root = new TreeNode(val);
            return;
        }
        TreeNode pre = null;
        TreeNode cur = root;

        while(cur != null) {
            pre = cur;
            if(cur.val < val) {
                cur = cur.right;
            }else if(cur.val > val){
                cur = cur.left;
            }else {//二叉搜索树中不要插入相同的数据(无意义)
                return;
            }
        }
        if(val < pre.val) {
            pre.left = new TreeNode(val);
        }else {
            pre.right = new TreeNode(val);
        }
    }
    //中序遍历
    public void inOrder(TreeNode root) {
        if(root == null) return;
        inOrder(root.left);
        System.out.print(root.val + " ");
        inOrder(root.right);
    }
    //删除
    public void remove(int val) {
        TreeNode cur = root;
        TreeNode pre = null;
        while(cur != null) {//找到了值为val的节点
            if(cur.val == val) {
                removeNode(pre,cur);
                return;
            }
            else if(cur.val < val) {
                pre = cur;
                cur = cur.right;
            }else {//cur.val > val
                pre = cur;
                cur = cur.left;
            }
        }
    }
    //删除值为val的cur节点
    private void removeNode(TreeNode pre,TreeNode cur) {
        if(cur.left == null) {
            if(cur == root) {
                root = cur.right;
            }else if(pre.left == cur){//cur在父节点左侧
                pre.left = cur.right;
            }else {//pre.right = cur --> //cur在父节点右侧
                pre.right = cur.right;
            }
        }else if(cur.right == null) {
            if(cur == root) {
                root = cur.left;
            }else if(pre.left == cur) {//cur在父节点左侧
                pre.left = cur.left;
            }else {//pre.right = cur --> //cur在父节点右侧
                pre.right = cur.left;
            }
        }else {//cur.left != null && cur.right != null
            TreeNode targetPre = cur;
            TreeNode target = cur.right;
            while(target.left != null) {//以cur的右子树的最小值举例-->右子树的左叶子节点
                //也可以是cur的左子树的最大值-->左子树的右叶子节点
                targetPre = target;
                target = target.left;
            }
            cur.val = target.val;
            //此时target的左侧一定为null-->target.left == null
            targetPre.left = target.right;//因为是寻找cur右子树的最小值
            //一定是target == targetPre.left
            //target == targetPre.right-->只会出现在寻找cur左子树的最大值
        }
     }
}

🔎结尾

欢迎各位点赞留言,如果有不懂可以在评论区探讨或者私信,希望和大家一起进步

  • 5
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 6
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值