二叉排序树的建立、查找、删除

二叉排序树又称为二叉查找树,具备以下性质:
①若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值;
②若它的右子树不空,则右子树上所有结点的值均大于它的根结点的值;
③它的左、右子树也分别为二叉排序树;

二叉排序树使用链式存储,插入和删除的效率较好,又可以较高效的实现查找某个元素的功能。

中序遍历二叉排序树时能够得到一个有序序列。

测试二叉排序树.png

二叉排序树结点类

/**
* @author Shaw
* @version 创建时间:2018年12月9日 下午3:33:43 
* 类说明:二叉排序树结点类
 */
class BiTNode {
    //数据域
    private int data;
    //左右结点域
    private BiTNode lchild, rchild;
    
    public BiTNode(int data) {
        this.data = data;
    }
    public int getData() {
        return data;
    }
    public void setData(int data) {
        this.data = data;
    }
    public BiTNode getLchild() {
        return lchild;
    }
    public void setLchild(BiTNode lchild) {
        this.lchild = lchild;
    }
    public BiTNode getRchild() {
        return rchild;
    }
    public void setRchild(BiTNode rchild) {
        this.rchild = rchild;
    }
    
    @Override
    public String toString() {
        return "BiTNode [data=" + data + ", lchild=" + lchild + ", rchild=" + rchild + "]";
    }   
}

二叉排序树核心类

二叉排序树的建立就是不断执行插入操作的过程。

二叉排序树的删除分为三种情况讨论:
①当删除结点仅有左子树时,只需将此结点的左孩子替换它自己,就相当于删除了该结点。
②当删除结点仅有右子树时,只需将此结点的右孩子替换它自己即可。
③当删除结点左右子树都不为空时,可以在左子树中找到小于但最接近该值的结点替换它,即找到中序遍历中的前驱;也可以在右子树中找到大于但最接近该值的结点替换,即中序遍历中的后驱。
本例中采用的是前驱替换。

/**
 * 
 * @author Shaw
 * @version 创建时间:2018年12月9日 下午3:34:11 
 * 类说明:二叉排序树类
 */
class BinarySortTree {
    public BiTNode root;

    public BinarySortTree() {
        root = null;
    }

    // 中序遍历
    public void InOrderTraverse(BiTNode node) {
        if (node == null) {
            return;
        }
        InOrderTraverse(node.getLchild());
        System.out.print(node.getData() + " ");
        InOrderTraverse(node.getRchild());
    }

    // 二叉排序树查找
    public BiTNode search_BST(int key) {
        BiTNode current = root;
        while (current != null) {
            // 查找成功返回对应结点
            if (key == current.getData()) {
                return current;
            } else if (key < current.getData()) {
                current = current.getLchild();
            } else {
                current = current.getRchild();
            }
        }
        return null;
    }

    // 二叉排序树插入
    public void insert_BST(int key) {
        // 空树情况
        if (root == null) {
            root = new BiTNode(key);
            return;
        }
        // 结点已在树中存在时
        if (search_BST(key) != null) {
            return;
        }
        BiTNode node = new BiTNode(key);
        BiTNode current = root;
        BiTNode parent = null;
        // 循环获取待插入结点的父结点位置
        while (current != null) {
            parent = current;
            if (key < current.getData()) {
                current = current.getLchild();
            } else if (key > current.getData()) {
                current = current.getRchild();
            } else {
                break;
            }
        }
        // 判断插入的是左结点还是右结点
        if (key < parent.getData()) {
            parent.setLchild(node);
        } else {
            parent.setRchild(node);
        }
    }

    // 二叉排序树删除
    public boolean delete_BST(int key) {
        // current结点指向待删除的结点
        BiTNode current = root;
        // parent结点指向待删除结点的父节点
        BiTNode parent = null;
        // 通过循环查找确定current和parent结点
        while (current != null) {
            // 待删除的结点的值小于根结点的值,查找左子树
            if (key < current.getData()) {
                parent = current;
                current = current.getLchild();
            }
            // 待删除的结点的值小于根结点的值,查找右子树
            else if (key > current.getData()) {
                parent = current;
                current = current.getRchild();
            }
            // 查找到结点后退出
            else {
                break;
            }
        }
        // 空树的情况
        if (current == null) {
            return false;
        }
        // 右子树为空的情况
        // 待删除结点的左结点"继承"该结点的位置
        if (current.getRchild() == null) {
            // 不是空树的情况
            if (parent != null) {
                // 待删除的结点是父节点的左结点
                if (parent.getLchild() == current) {
                    // 将待删除结点的左结点设置为父结点的左结点
                    parent.setLchild(current.getLchild());
                } else {
                    // 待删除的结点是父结点的右结点时将该结点的左结点设置为父结点的右结点
                    parent.setRchild(current.getLchild());
                }
            } else {
                // 空树时将左结点赋值给根结点
                root = current.getLchild();
            }
        }
        // 左子树为空的情况
        // 待删除结点的右结点"继承"该结点的位置
        else if (current.getLchild() == null) {
            if (parent != null) {
                if (parent.getLchild() == current) {
                    parent.setLchild(current.getLchild());
                } else {
                    parent.setRchild(current.getRchild());
                }
            } else {
                root = current.getRchild();
            }
        }
        // 左右子树均不为空的情况
        // 在二叉排序树中序中选择该结点的前驱或者后继替换该结点,
        // 也就是选择比该结点小或者比它大的最接近的两个数中的一个,本例选择的是比该结点小的结点
        else {
            // 用于替换的结点
            BiTNode replaceNode = current.getLchild();
            // 替换结点的父节点,初始化为待删除的结点
            BiTNode replaceParentNode = current;
            // 用于替换的结点存在右子结点时,因为右结点大于根结点的值,所以右子结点更接近被删除的结点
            while (replaceNode.getRchild() != null) {
                replaceParentNode = replaceNode;
                replaceNode = replaceNode.getRchild();
            }
            // 替换结点不存在右子结点时
            // 相等时由于使用的是前驱值代替,所以需要补充的是左子结点的值
            if (replaceParentNode == current) {
                replaceParentNode.setLchild(replaceNode.getLchild());
            }
            // replaceParentNode与current不相等时
            // replaceNode肯定是大于replaceParentNode的值的,所以需要补充的是右子结点的值
            else {
                replaceParentNode.setRchild(replaceNode.getLchild());
            }
            // 替换对应的值
            current.setData(replaceNode.getData());
        }
        return true;
    }
}

测试程序:

public class BinarySortTreeMain {

    public static void main(String[] args) {
        // TODO Auto-generated method stub
        int[] a = { 62, 88, 58, 47, 35, 73, 51, 99, 37, 93, 36, 39, 42, 62 };
        BinarySortTree binarySortTree = new BinarySortTree();
        System.out.println("原始序列:");
        for (int i = 0; i < a.length; i++) {
            System.out.print(a[i] + " ");
            binarySortTree.insert_BST(a[i]);
        }
        System.out.println("\n二叉排序后的序列:");
        binarySortTree.InOrderTraverse(binarySortTree.root);
        System.out.print("\n查找37:");
        BiTNode node = binarySortTree.search_BST(37);
        if (node != null) {
            System.out.println(true);
        } else {
            System.out.println(false);
        }
        System.out.print("查找22:");
        node = binarySortTree.search_BST(22);
        if (node != null) {
            System.out.println(true);
        } else {
            System.out.println(false);
        }
        System.out.print("删除47:");
        System.out.println(binarySortTree.delete_BST(47));
        System.out.println("删除47后的序列:");
        binarySortTree.InOrderTraverse(binarySortTree.root);
        System.out.print("\n删除22:");
        System.out.println(binarySortTree.delete_BST(22));
    }
}

测试结果:

二叉排序树测试结果图.png

总结

二叉排序树以链接方式存储,保持了链接存储结构在执行插入或删除操作时不用移动元素的优点,插入删除的时间性能较好。对于二叉排序树的查找,比较次数等于给定值的结点在二叉排序树的层数。最少为1次,最多也不会超过树的深度。

缺点:当本例中的数组是{35,36,37,39,42,47,51,58,73,88,93,99}时,即数组元素从小到大有序时,此时的二叉排序树如图所示是一棵右斜树。当同样查找99结点时,测试例子需要2次比较,本例却需要12次比较,二者差异很大。也就是说,二叉排序树的查找性能主要取决于二叉排序树的形状,但本例中的二叉排序树的形状是由给定数组来构造的,即形状是不确定的。为此,我们需要构造平衡二叉树来解决这个问题。

右斜树.png

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

推荐阅读更多精彩内容