服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - Java教程 - 利用java实现二叉搜索树

利用java实现二叉搜索树

2021-09-08 10:47\(^o^)/kūn Java教程

这篇文章主要介绍了利用java实现二叉搜索树,文中有非常详细的代码示例,对正在学习java的小伙伴们有非常好的帮助,需要的朋友可以参考下

二叉搜索树的定义

  • 它是一颗二叉树
  • 任一节点的左子树上的所有节点的值一定小于该节点的值
  • 任一节点的右子树上的所有节点的值一定大于该节点的值

利用java实现二叉搜索树

特点: 二叉搜索树的中序遍历结果是有序的(升序)!

实现一颗二叉搜索树

  • 实现二叉搜索树,将实现插入,删除,查找三个方面
  • 二叉搜索树的节点是不可以进行修改的,如果修改,则可能会导致搜索树的错误

二叉搜索树的定义类

  • 二叉搜索树的节点类 —— class node
  • 二叉搜索树的属性:要找到一颗二叉搜索树只需要知道这颗树的根节点。
?
1
2
3
4
5
6
7
8
9
10
11
12
13
public class bst {
    static class node {
        private int key;
        private node left;
        private node right;
 
        public node(int key) {
            this.key = key;
        }
    }
 
    private node root;//bst的根节点
}

二叉搜索树的查找

  • 二叉搜索树的查找思路:
  • ①如果要查找的值等于当前节点的值,那么,就找到了
  • ②如果要查找的值小于当前节点的值,那么,就往当前节点的左子树走
  • ③如果要查找的值大于当前节点的值,那么,就往当前节点的右子树走
  • 最终,如果走到空了还没有找到,就说明不存在这个key
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
/**
 * 查找是否存在节点
 *
 * 思路:根据二叉排序树的特点:
 * ①如果要查找的值小于当前节点的值,那么,就往当前节点的左子树走
 * ②如果要查找的值大于当前节点的值,那么,就往当前节点的右子树走
 *
 * @param key 带查找的key
 * @return boolean是否存在
 */
public boolean find(int key) {
    node cur = root;
    while (cur != null) {
        if (key < root.key) {
            cur = cur.left;
        } else if (key > root.key) {
            cur = cur.right;
        } else {
            return true;
        }
    }
    return false;
}

二叉搜索树的插入

  • 二叉搜索树的插入思路:
  • 思路和查找一样的,只是我们这次要进行的是插入操作,那么我们还需要一个parent节点,来时刻记录当前节点的双亲节点即:
  • ①如果要插入的值等于当前节点的值,那么,无法插入(不可出现重复的key
  • ②如果要插入的值小于当前节点的值,那么,就往当前节点的左子树走
  • ③如果要插入的值大于当前节点的值,那么,就往当前节点的右子树走
  • 最终,如果走到空了,就说明不存在重复的key,只要往双亲节点的后面插就好了,就是合适的位置,具体往左边还是右边插入,需要比较待插入节点的keyparentkey
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
/**
 * 往二叉树中插入节点
 *
 * 思路如下:
 *
 * @param key 待插入的节点
 */
public void insert(int key) {
    if (root == null) { //如果是空树,那么,直接插入
        root = new node(key);
        return;
    }
 
    node cur = root;
    node parent = null; //parent 为cur的父节点
    while (true) {
        if (cur == null) { //在遍历过程中,找到了合适是位置,就指针插入(没有重复节点)
            if (parent.key < key) {
                parent.right = new node(key);
            } else {
                parent.left = new node(key);
            }
            return;
        }
 
        if (key < cur.key) {
            parent = cur;
            cur = cur.left;
        } else if (key > cur.key) {
            parent = cur;
            cur = cur.right;
        } else {
            throw new runtimeexception("插入失败,已经存在key");
        }
    }
}

二叉搜索树的删除

  • 二叉搜索树的删除思路:(详细的思路看注释
  • 首先,需要先找到是否存在key节点,如果存在,则删除,如果不存在则删除错误
  • 对于,如果存在,则分为三种情况:
  • ①要删除的节点,没有左孩子

ⅰ:要删除的节点为根节点:root = delete.right;
ⅱ:要删除的节点为其双亲节点的左孩子:parent.left = delete.right;
ⅲ:要删除的节点为其双亲节点的右孩子:parent.right = delete.right;

  • ②要删除的节点,没有右孩子

ⅰ:要删除的节点为根节点:root = delete.left;
ⅱ:要删除的节点为其双亲节点的左孩子:parent.left = delete.left;
ⅲ:要删除的节点为其双亲节点的右孩子:parent.right = delete.left;

  • ③要删除的节点,既有左孩子又有右孩子:

此时我们需要找到整颗二叉树中第一个大于待删除节点的节点,然后替换他俩的值,最后,把找到的节点删除
ⅰ:找到的节点的双亲节点为待删除的节点:delete.key = find.key; findparent.right = find.right;
ⅱ:找到的节点的双亲节点不是待删除的节点:delete.key = find.key; findparent.left = find.right;

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
/**
 * 删除树中节点
 *
 * 思路如下:
 *
 * @param key 待删除的节点
 */
public void remove(int key) {
    if (root == null) {
        throw new runtimeexception("为空树,删除错误!");
    }
    node cur = root;
    node parent = null;
    //查找是否key节点的位置
    while (cur != null) {
        if (key < cur.key) {
            parent = cur;
            cur = cur.left;
        } else if (key > cur.key) {
            parent = cur;
            cur = cur.right;
        } else {
            break;
        }
    }
    if (cur == null) {
        throw new runtimeexception("找不到key,输入key不合法");
    }
 
    //cur 为待删除的节点
    //parent 为待删除的节点的父节点
    /*
         * 情况1:如果待删除的节点没有左孩子
         * 其中
         * ①待删除的节点有右孩子
         * ②待删除的节点没有右孩子
         * 两种情况可以合并
         */
    if (cur.left == null) {
        if (cur == root) { //①如果要删除的是根节点
            root = cur.right;
        } else if (cur == parent.left) { //②如果要删除的是其父节点的左孩子
            parent.left = cur.right;
        } else { //③如果要删除的节点为其父节点的右孩子
            parent.right = cur.right;
        }
    }
    /*
         * 情况2:如果待删除的节点没有右孩子
         *
         * 其中:待删除的节点必定存在左孩子
         */
    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;
        }
    }
    /*
        * 情况3:如果待删除的节点既有左孩子又有右孩子
        *
        * 思路:
        * 因为是排序二叉树,要找到整颗二叉树第一个大于该节点的节点,只需要,先向右走一步,然后一路往最左走就可以找到了
        * 因此:
        * ①先向右走一步
        * ②不断向左走
        * ③找到第一个大于待删除的节点的节点,将该节点的值,替换到待删除的节点
        * ④删除找到的这个节点
        * ⑤完成删除
        *
         */
    else {
        node nextparent = cur; //定义父节点,初始化就是待删除的节点
        node next = cur.right; //定义next为当前走到的节点,最终目的是找到第一个大于待删除的节点
        while (next.left != null) {
            nextparent = next;
            next = next.left;
        }
        cur.key = next.key; //找到之后,完成值的替换
        if (nextparent == cur) { //此时的父节点就是待删除的节点,那么说明找到的节点为父节点的右孩子(因为此时next只走了一步)
            nextparent.right = next.right;
        } else { //此时父节点不是待删除的节点,即next确实往左走了,且走到了头.
            nextparent.left = next.right;
        }
    }
 
}

到此这篇关于利用java实现二叉搜索树的文章就介绍到这了,更多相关java二叉搜索树内容请搜索服务器之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持服务器之家!

原文链接:https://blog.csdn.net/qq_44025641/article/details/115712153

延伸 · 阅读

精彩推荐
  • Java教程Java BufferWriter写文件写不进去或缺失数据的解决

    Java BufferWriter写文件写不进去或缺失数据的解决

    这篇文章主要介绍了Java BufferWriter写文件写不进去或缺失数据的解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望...

    spcoder14552021-10-18
  • Java教程小米推送Java代码

    小米推送Java代码

    今天小编就为大家分享一篇关于小米推送Java代码,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧...

    富贵稳中求8032021-07-12
  • Java教程Java使用SAX解析xml的示例

    Java使用SAX解析xml的示例

    这篇文章主要介绍了Java使用SAX解析xml的示例,帮助大家更好的理解和学习使用Java,感兴趣的朋友可以了解下...

    大行者10067412021-08-30
  • Java教程升级IDEA后Lombok不能使用的解决方法

    升级IDEA后Lombok不能使用的解决方法

    最近看到提示IDEA提示升级,寻思已经有好久没有升过级了。升级完毕重启之后,突然发现好多错误,本文就来介绍一下如何解决,感兴趣的可以了解一下...

    程序猿DD9332021-10-08
  • Java教程xml与Java对象的转换详解

    xml与Java对象的转换详解

    这篇文章主要介绍了xml与Java对象的转换详解的相关资料,需要的朋友可以参考下...

    Java教程网2942020-09-17
  • Java教程Java8中Stream使用的一个注意事项

    Java8中Stream使用的一个注意事项

    最近在工作中发现了对于集合操作转换的神器,java8新特性 stream,但在使用中遇到了一个非常重要的注意点,所以这篇文章主要给大家介绍了关于Java8中S...

    阿杜7472021-02-04
  • Java教程20个非常实用的Java程序代码片段

    20个非常实用的Java程序代码片段

    这篇文章主要为大家分享了20个非常实用的Java程序片段,对java开发项目有所帮助,感兴趣的小伙伴们可以参考一下 ...

    lijiao5352020-04-06
  • Java教程Java实现抢红包功能

    Java实现抢红包功能

    这篇文章主要为大家详细介绍了Java实现抢红包功能,采用多线程模拟多人同时抢红包,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙...

    littleschemer13532021-05-16