二叉查找树的Java实现

最近在闲看博客时看到一篇专门写红黑树的实现原理,以Java的TreeMap为例讲解,写的很不错,仔细看下来发现很多地方不是很理解,毕竟没有对树的理解并没有很深,所以决定一步一步的将与树相关的扩展实现都了解一遍,沿着下面的学习路线开始,大家也可以参考以下。

  • 树的基本知识
  • 二叉树的知识
  • 二叉查找树
  • 平衡二叉树
  • 红黑树
  • B树,B-树,B+树

附上上面的将红黑树的blog:史上最清晰的红黑树讲解

树的基本概念

Tree是n(n≥0)个结点的有限集。在任意一棵非空树中:(1)有且仅有一个特定的被称为根root的结点;(2)当n>1时,其余结点可分为m(m>0)个互不相交的有限集T1,T2,…,Tm,其中每一个集合本身又是一棵树,并且称为根的子树SubTree

  1. 结点拥有的子树数称为结点的度(Degree)。度为0的结点称为叶子(Leaf)或终端结点。度不为0的结点称为非终端结点或分支结点。
  2. 树的度是树内各结点的度的最大值。
  3. 结点的子树的根称为该结点的孩子(Child),相应地,该结点称为孩子的双亲(Parent)。
  4. 结点的层次(Level)是从根结点开始计算起,根为第一层,根的孩子为第二层,依次类推。树中结点的最大层次称为树的深度(Depth)或高度。
  5. 如果将树中结点的各子树看成从左至右是有次序的(即不能互换),则称该树为有序树,否则称为无序树。

二叉树的基本概念

二叉树(Binary Tree)的特点是每个结点至多具有两棵子树(即在二叉树中不存在度大于2的结点),并且子树之间有左右之分。

二叉树的性质:

  1. 在二叉树的第i层上至多有2i-1个结点(i≥1)。

  2. 深度为k的二叉树至多有2k-1个结点(k≥1)。

  3. 对任何一棵二叉树,如果其终端结点数为n0,度为2的结点数为n2,则n0=n2+1。

  4. 具有n个结点的完全二叉树的深度为不大于log2n的最大整数加1。

  5. 如果对一棵有n个结点的完全二叉树的结点按层序编号(从第1层到最后一层,每层从左到右),则对任一结点i(1≤i≤n),有

    a、如果i=1,则结点i是二叉树的根,无双亲;如果i>1,则其双亲是结点x(其中x是不大于i/2的最大整数)。

    b、如果2i>n,则结点i无左孩子(结点i为叶子结点);否则其左孩子是结点2i。

    c、如果2i+1>n,则结点i无右孩子;否则其右孩子是结点2i+1。

二叉查找树

二叉查找树(BinarySearch Tree,也叫二叉搜索树,或称二叉排序树Binary Sort Tree)或者是一棵空树,或者是具有下列性质的二叉树:

  1. 若它的左子树不为空,则左子树上所有结点的值均小于它的根结点的值;
  2. 若它的右子树不为空,则右子树上所有结点的值均大于它的根结点的值;
  3. 它的左、右子树也分别为二叉查找树。

结点定义:

public class TreeNode<T> {

    /**
     * 结点的值
     */
    private T value;

    /**
     * 左结点
     */
    private TreeNode<T> left;
    
    /**
     * 右结点
     */
    private TreeNode<T> right;
    
    /**
     * 父亲结点
     */
    private TreeNode<T> parent;

    /**
     * 频率
     */
    private int freq;

}

插入

根据二叉查找树的性质,插入一个节点的时候,如果根节点为空,就此节点作为根节点,如果根节点不为空,就要先和根节点比较,如果比根节点的值小,就插入到根节点的左子树中,如果比根节点的值大就插入到根节点的右子树中,如此递归下去,找到插入的位置。重复节点的插入用值域中的freq标记。如图2是一个插入的过程。

插入过程图1

二叉查找树的时间复杂度要看这棵树的形态,如果比较接近一一棵完全二叉树,那么时间复杂度在O(logn)左右,如果遇到如图3这样的二叉树的话,那么时间复杂度就会恢复到线性的O(n)了。

插入过程图2

平衡二叉树会很好的解决如图3这种情况。

核心代码如下:

private boolean insert(SearchNode<T> curr, SearchNode<T> insertNode, SearchNode<T> parent, boolean currIsLeft) {
    if (curr == null) {
        curr = insertNode;
        if (currIsLeft) {
            parent.setLeft(curr);
        } else {
            parent.setRight(curr);
        }
    } else {
        int result = curr.getValue().compareTo(insertNode.getValue());
        // 如果当前值大于插入的值
        if (result > 0) {
            return insert((SearchNode<T>)curr.getLeft(), insertNode, curr, true);
        } else if (result < 0) {
            return insert((SearchNode<T>)curr.getRight(), insertNode, curr, false);
        }else {
            curr.freq++;
        }
    }
    return true;
}

查找

在二叉查找树中查找x的过程如下:

  1. 若二叉树是空树,则查找失败。
  2. 若x等于根结点的数据,则查找成功,否则。
  3. 若x小于根结点的数据,则递归查找其左子树,否则。
  4. 递归查找其右子树。

核心代码如下:

protected TreeNode<T> find0(TreeNode<T> node, T value) {
    if (node == null) {
        return null;
    }
    int result = node.getValue().compareTo(value);
    if (result > 0) {
        return find0(node.getLeft(), value);
    } else if (result < 0) {
        return find0(node.getRight(), value);
    }
    return node;
}

删除

删除会麻烦一点,如果是叶子节点的话,直接删除就可以了。如果只有一个孩子的话,就让它的父亲指向它的儿子,然后删除这个节点。图4显示了一棵初始树和4节点被删除后的结果。先用一个临时指针指向4节点,再让4节点的地址指向它的孩子,这个时候2节点的右儿子就变成了3节点,最后删除临时节点指向的空间,也就是4节点。

删除过程图1

删除有两个儿子的节点会比较复杂一些。一般的删除策略是用其右子树最小的数据代替该节点的数据并递归的删除掉右子树中最小数据的节点。因为右子树中数据最小的节点肯定没有左儿子,所以删除的时候容易一些。图5显示了一棵初始树和2节点被删除后的结果。首先在2节点的右子树中找到最小的节点3,然后把3的数据赋值给2节点,这个时候2节点的数据变为3,然后的工作就是删除右子树中的3节点了,采用递归删除。

删除过程图2

我们发现对2节点右子树的查找进行了两遍,第一遍找到最小节点并赋值,第二遍删除这个最小的节点,这样的效率并不是很高。你能不能写出只查找一次就可以实现赋值和删除两个功能的函数呢?

如果删除的次数不是很多的话,有一种删除的方法会比较快一点,名字叫懒惰删除法:当一个元素要被删除时,它仍留在树中,只是多了一个删除的标记。这种方法的优点是删除那一步的时间开销就可以避免了,如果重新插入删除的节点的话,插入时也避免了分配空间的时间开销。缺点是树的深度会增加,查找的时间复杂度会增加,插入的时间可能会增加。

核心代码如下:

protected void deleteNode(TreeNode<T> deleteNodeParent, TreeNode<T> deleteNode) {
    if (deleteNodeParent == null) {
        // 左右子树都为空
        if (deleteNode.getLeft() == null && deleteNode.getRight() == null) {
            root = null;
        } else if (deleteNode.getLeft() == null || deleteNode.getRight() == null) {
            // 存在左子树或右子树
            if (deleteNode.getLeft() != null) {
                root = deleteNode.getLeft();
            } else {
                root = deleteNode.getRight();
            }
        } else {
            // 左右子树都不为空
            TreeNode<T> temp = deleteNode;
            TreeNode<T> rightLeft = deleteNode.getRight();
            while (rightLeft.getLeft() != null) {
                temp = rightLeft;
                rightLeft = rightLeft.getLeft();
            }
            if(temp == deleteNode) {
                deleteNode.setRight(rightLeft.getRight());
            }else {
                temp.setLeft(rightLeft.getRight());
            }
            deleteNode.setValue(rightLeft.getValue());  
        }
    } else {
        // 左右子树都为空
        if (deleteNode.getLeft() == null && deleteNode.getRight() == null) {
            // 根结点
            if (deleteNodeParent.getLeft() == deleteNode) {
                deleteNodeParent.setLeft(null);
            } else {
                deleteNodeParent.setRight(null);
            }
        } else if (deleteNode.getLeft() == null || deleteNode.getRight() == null) {
            // 存在左子树或右子树
            if (deleteNode.getLeft() != null) {
                if (deleteNodeParent.getLeft() == deleteNode) {
                    // 如果待删除结点是左结点,且其存在左结点
                    deleteNodeParent.setLeft(deleteNode.getLeft());
                } else {
                    // 如果待删除结点是左结点,且其存在右结点
                    deleteNodeParent.setRight(deleteNode.getLeft());
                }
            } else {
                if (deleteNodeParent.getRight() == deleteNode) {
                    deleteNodeParent.setRight(deleteNode.getRight());
                } else {
                    deleteNodeParent.setLeft(deleteNode.getRight());
                }
            }
        } else {
            // 左右子树都不为空
            TreeNode<T> temp = deleteNode;
            TreeNode<T> rightLeft = deleteNode.getRight();
            while (rightLeft.getLeft() != null) {
                temp = rightLeft;
                rightLeft = rightLeft.getLeft();
            }
            if(temp == deleteNode) {
                deleteNode.setRight(rightLeft.getRight());
            }else {
                temp.setLeft(rightLeft.getRight());
            }
            deleteNode.setValue(rightLeft.getValue());  
        }
    }

}

具体实现代码请查看:https://github.com/mastery001/study/blob/master/study-datastruct/src/main/java/tree/search/BinarySearchTree.java

参考资料:

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,189评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,577评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,857评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,703评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,705评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,620评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,995评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,656评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,898评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,639评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,720评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,395评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,982评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,953评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,195评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,907评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,472评论 2 342

推荐阅读更多精彩内容

  • B树的定义 一棵m阶的B树满足下列条件: 树中每个结点至多有m个孩子。 除根结点和叶子结点外,其它每个结点至少有m...
    文档随手记阅读 13,149评论 0 25
  • 数据结构与算法--从平衡二叉树(AVL)到红黑树 上节学习了二叉查找树。算法的性能取决于树的形状,而树的形状取决于...
    sunhaiyu阅读 7,637评论 4 32
  • 目录 0.树0.1 一般树的定义0.2 二叉树的定义 1.查找树ADT 2.查找树的实现2.1 二叉查找树2.2 ...
    王侦阅读 7,112评论 0 3
  • 树的概述 树是一种非常常用的数据结构,树与前面介绍的线性表,栈,队列等线性结构不同,树是一种非线性结构 1.树的定...
    Jack921阅读 4,432评论 1 31
  • 四、树与二叉树 1. 二叉树的顺序存储结构 二叉树的顺序存储就是用数组存储二叉树。二叉树的每个结点在顺序存储中都有...
    MinoyJet阅读 1,499评论 0 7