Java深入了解数据结构之二叉搜索树增 插 删 创详解

作者:/少司命 时间:2023-02-14 08:08:00 

①概念

二叉搜索树又称二叉排序树,它或者是一棵空树**,或者是具有以下性质的二叉树:

若它的左子树不为空,则左子树上所有节点的值都小于根节点的值

若它的右子树不为空,则右子树上所有节点的值都大于根节点的值

它的左右子树也分别为二叉搜索树

Java深入了解数据结构之二叉搜索树增 插 删 创详解

②操作-查找

二叉搜索树的查找类似于二分法查找

Java深入了解数据结构之二叉搜索树增 插 删 创详解

Java深入了解数据结构之二叉搜索树增 插 删 创详解

Java深入了解数据结构之二叉搜索树增 插 删 创详解


public Node search(int key) {
       Node cur = root;
       while (cur != null) {
           if(cur.val == key) {
               return cur;
           }else if(cur.val < key) {
               cur = cur.right;
           }else {
               cur = cur.left;
           }
       }
       return null;
   }

③操作-插入

Java深入了解数据结构之二叉搜索树增 插 删 创详解

Java深入了解数据结构之二叉搜索树增 插 删 创详解

Java深入了解数据结构之二叉搜索树增 插 删 创详解

Java深入了解数据结构之二叉搜索树增 插 删 创详解


 public boolean insert(int key) {
       Node node = new Node(key);
       if(root == null) {
           root = node;
           return true;
       }

Node cur = root;
       Node parent = null;

while(cur != null) {
           if(cur.val == key) {
               return false;
           }else if(cur.val < key) {
               parent = cur;
               cur = cur.right;
           }else {
               parent = cur;
               cur = cur.left;
           }
       }
       //parent
       if(parent.val > key) {
           parent.left = node;
       }else {
           parent.right = node;
       }

return true;
   }

④操作-删除

删除操作较为复杂,但理解了其原理还是比较容易

设待删除结点为 cur, 待删除结点的双亲结点为 parent

1. cur.left == null

1. cur 是 root,则 root = cur.right

2. cur 不是 root,cur 是 parent.left,则 parent.left = cur.right

3. cur 不是 root,cur 是 parent.right,则 parent.right = cur.right

Java深入了解数据结构之二叉搜索树增 插 删 创详解

Java深入了解数据结构之二叉搜索树增 插 删 创详解

Java深入了解数据结构之二叉搜索树增 插 删 创详解

Java深入了解数据结构之二叉搜索树增 插 删 创详解

2. cur.right == null

1. cur 是 root,则 root = cur.left

2. cur 不是 root,cur 是 parent.left,则 parent.left = cur.left

3. cur 不是 root,cur 是 parent.right,则 parent.right = cur.left

第二种情况和第一种情况相同,只是方向相反,这里不再画图

3. cur.left != null && cur.right != null

需要使用替换法进行删除,即在它的右子树中寻找中序下的第一个结点(关键码最小),用它的值填补到被删除节点中,再来处理该结点的删除问题

当我们在左右子树都不为空的情况下进行删除,删除该节点会破坏树的结构,因此用替罪羊的方法来解决,实际删除的过程还是上面的两种情况,这里还是用到了搜索二叉树的性质

Java深入了解数据结构之二叉搜索树增 插 删 创详解

Java深入了解数据结构之二叉搜索树增 插 删 创详解

Java深入了解数据结构之二叉搜索树增 插 删 创详解


public void remove(Node parent,Node cur) {
       if(cur.left == null) {
           if(cur == root) {
               root = cur.right;
           }else if(cur == parent.left) {
               parent.left = cur.right;
           }else {
               parent.right = cur.right;
           }
       }else if(cur.right == null) {
           if(cur == root) {
               root = cur.left;
           }else if(cur == parent.left) {
               parent.left = cur.left;
           }else {
               parent.right = cur.left;
           }
       }else {
           Node targetParent =  cur;
           Node target = cur.right;
           while (target.left != null) {
               targetParent = target;
               target = target.left;
           }
           cur.val = target.val;
           if(target == targetParent.left) {
               targetParent.left = target.right;
           }else {
               targetParent.right = target.right;
           }
       }
   }

public void removeKey(int key) {
       if(root == null) {
           return;
       }
       Node cur = root;
       Node parent = null;
       while (cur != null) {
           if(cur.val == key) {
               remove(parent,cur);
               return;
           }else if(cur.val < key){
               parent = cur;
               cur = cur.right;
           }else {
               parent = cur;
               cur = cur.left;
           }
       }
   }

⑤性能分析

插入和删除操作都必须先查找,查找效率代表了二叉搜索树中各个操作的性能。

对有n个结点的二叉搜索树,若每个元素查找的概率相等,则二叉搜索树平均查找长度是结点在二叉搜索树的深度 的函数,即结点越深,则比较次数越多。

但对于同一个关键码集合,如果各关键码插入的次序不同,可能得到不同结构的二叉搜索树:

最优情况下,二叉搜索树为完全二叉树,其平均比较次数为:

Java深入了解数据结构之二叉搜索树增 插 删 创详解

最差情况下,二叉搜索树退化为单支树,其平均比较次数为:

Java深入了解数据结构之二叉搜索树增 插 删 创详解

⑥完整代码


public class TextDemo {

public static class Node {
       public int val;
       public Node left;
       public Node right;

public Node (int val) {
           this.val = val;
       }
   }

public Node root;

/**
    * 查找
    * @param key
    */
   public Node search(int key) {
       Node cur = root;
       while (cur != null) {
           if(cur.val == key) {
               return cur;
           }else if(cur.val < key) {
               cur = cur.right;
           }else {
               cur = cur.left;
           }
       }
       return null;
   }

/**
    *
    * @param key
    * @return
    */
   public boolean insert(int key) {
       Node node = new Node(key);
       if(root == null) {
           root = node;
           return true;
       }

Node cur = root;
       Node parent = null;

while(cur != null) {
           if(cur.val == key) {
               return false;
           }else if(cur.val < key) {
               parent = cur;
               cur = cur.right;
           }else {
               parent = cur;
               cur = cur.left;
           }
       }
       //parent
       if(parent.val > key) {
           parent.left = node;
       }else {
           parent.right = node;
       }

return true;
   }

public void remove(Node parent,Node cur) {
       if(cur.left == null) {
           if(cur == root) {
               root = cur.right;
           }else if(cur == parent.left) {
               parent.left = cur.right;
           }else {
               parent.right = cur.right;
           }
       }else if(cur.right == null) {
           if(cur == root) {
               root = cur.left;
           }else if(cur == parent.left) {
               parent.left = cur.left;
           }else {
               parent.right = cur.left;
           }
       }else {
           Node targetParent =  cur;
           Node target = cur.right;
           while (target.left != null) {
               targetParent = target;
               target = target.left;
           }
           cur.val = target.val;
           if(target == targetParent.left) {
               targetParent.left = target.right;
           }else {
               targetParent.right = target.right;
           }
       }
   }

public void removeKey(int key) {
       if(root == null) {
           return;
       }
       Node cur = root;
       Node parent = null;
       while (cur != null) {
           if(cur.val == key) {
               remove(parent,cur);
               return;
           }else if(cur.val < key){
               parent = cur;
               cur = cur.right;
           }else {
               parent = cur;
               cur = cur.left;
           }
       }
   }

}

来源:https://blog.csdn.net/qq_50156012/article/details/122009098

标签:Java,二叉搜索树,数据结构
0
投稿

猜你喜欢

  • C#中缓存System.Web.Caching用法总结

    2021-09-05 04:41:11
  • Java基础高级综合练习题扑克牌的创建

    2023-09-08 06:56:19
  • SpringCloud Feign 服务调用的实现

    2023-09-18 11:07:35
  • c# 实现自动扫雷

    2021-09-01 09:25:58
  • JAVA MyBatis入门学习过程记录

    2022-04-23 13:49:24
  • Java的异常体系以及File类构造方法详解

    2021-09-05 20:06:41
  • Android 自定义gradle property详解及实例代码

    2022-05-08 15:17:49
  • C#控制台模拟电梯工作原理

    2021-08-06 02:35:33
  • Android微信抢红包功能的实现原理浅析

    2023-11-20 19:14:40
  • Java贪吃蛇游戏完善版

    2023-04-12 03:07:53
  • Java ThreadPoolExecutor线程池有关介绍

    2022-11-21 02:03:20
  • java算法之Math.random()随机概率玩法实例演示

    2023-11-28 23:32:17
  • Java 8 开发的 Mybatis 注解代码生成工具

    2023-01-02 19:53:44
  • javafx实现五子棋游戏

    2022-02-01 07:00:01
  • Java基于栈方式解决汉诺塔问题实例【递归与非递归算法】

    2023-04-10 08:05:02
  • Opencv光流运动物体追踪详解

    2023-06-21 11:55:31
  • SpringBoot-RestTemplate实现调用第三方API的方式

    2022-12-29 09:49:56
  • c# 实现获取汉字十六进制Unicode编码字符串的实例

    2023-03-21 11:22:35
  • 利用logback 设置不同包下的日志级别

    2022-08-11 20:05:43
  • C#多线程与跨线程访问界面控件的方法

    2023-11-25 12:00:29
  • asp之家 软件编程 m.aspxhome.com