二叉平衡树

全图基础解释参考链接:http://btechsmartclass.com/data_structures/avl-trees.html

二叉平衡树:https://www.cnblogs.com/zhuwbox/p/3636783.html

前提:会写 求二叉树的深度

背景知识:

为什么需要二叉平衡树

答:因为二叉搜索树在理想状态下(也就是平衡树),查找的时间复杂度为log2n ,但是如果很不幸,

​ 插入的数据都是有序数据的话,那么会退化成O(n)的线性时间复杂度。因为几乎退化成了链!

线性:6次

平衡:3次 log6+1 = 3

总结:树的基本操作的时间复杂度几乎都与树的高度有关,那么减少树的高度,就可以降低查询的时间复杂度。

 我们知道,对于一般的二叉搜索树(Binary Search Tree),其期望高度(即为一棵平衡树时)为log2n,其各操作的时间复杂度(O(log2n))同时也由此而决定。但是,在某些极端的情况下(如在插入的序列是有序的时),二叉搜索树将退化成近似链或链,此时,其操作的时间复杂度将退化成线性的,即O(n)。我们可以通过随机化建立二叉搜索树来尽量的避免这种情况,但是在进行了多次的操作之后,由于在删除时,我们总是选择将待删除节点的后继代替它本身,这样就会造成总是右边的节点数目减少,以至于树向左偏沉。这同时也会造成树的平衡性受到破坏,提高它的操作的时间复杂度。

  例如:我们按顺序将一组数据1,2,3,4,5,6分别插入到一颗空二叉查找树和AVL树中,插入的结果如下图:

        

AVL树的插入

  由上图可知,同样的结点,由于插入方式不同导致树的高度也有所不同。特别是在带插入结点个数很多且正序的情况下,会导致二叉树的高度是O(N),而AVL树就不会出现这种情况,树的高度始终是O(lgN).高度越小,对树的一些基本操作的时间复杂度就会越小。这也就是我们引入AVL树的原因

  AVL树的操作基本和二叉查找树一样,这里我们关注的是两个变化很大的操作:插入和删除!

  我们知道,AVL树不仅是一颗二叉查找树,它还有其他的性质。如果我们按照一般的二叉查找树的插入方式可能会破坏AVL树的平衡性。同理,在删除的时候也有可能会破坏树的平衡性,所以我们要做一些特殊的处理,包括:单旋转和双旋转!

  AVL树的插入,单旋转的第一种情况---右旋:

  由上图可知:在插入之前树是一颗AVL树,而插入之后结点T的左右子树高度差的绝对值不再 < 1,此时AVL树的平衡性被破坏,我们要对其进行旋转。由上图可知我们是在结点T的左结点的左子树上做了插入元素的操作,我们称这种情况为左左情况,我们应该进行右旋转(只需旋转一次,故是单旋转)。具体旋转步骤是:

  T向右旋转成为L的右结点,同时,Y放到T的左孩子上。这样即可得到一颗新的AVL树,旋转过程图如下:

  左左情况的右旋举例:

  AVL树的插入,单旋转的第一种情况---左旋:

 

   由上图可知:在插入之前树是一颗AVL树,而插入之后结点T的左右子树高度差的绝对值不再 < 1,此时AVL树的平衡性被破坏,我们要对其进行旋转。由上图可知我们是在结点T的右结点的右子树上做了插入元素的操作,我们称这种情况为右右情况,我们应该进行左旋转(只需旋转一次,故事单旋转)。具体旋转步骤是:

   T向右旋转成为R的左结点,同时,Y放到T的左孩子上。这样即可得到一颗新的AVL树,旋转过程图如下:

 

  右右情况的左旋举例:

  以上就是插入操作时的单旋转情况!我们要注意的是:谁是T谁是L,谁是R还有谁是X,Y,Z!T始终是开始不平衡的左右子树的根节点。显然L是T的左结点,R是T的右节点。X、Y、Y是子树当然也可以为NULL.NULL归NULL,但不能破坏插入时我上面所说的左左情况或者右右情况。

  AVL树的插入,双旋转的第一种情况---左右(先左后右)旋:

由  上图可知,我们在T结点的左结点的右子树上插入一个元素时,会使得根为T的树的左右子树高度差的绝对值不再 < 1,如果只是进行简单的右旋,得到的树仍然是不平衡的。我们应该按照如下图所示进行二次旋转:

  

  左右情况的左右旋转实例:

  AVL树的插入,双旋转的第二种情况---右左(先右后左)旋:

  由上图可知,我们在T结点的右结点的左子树上插入一个元素时,会使得根为T的树的左右子树高度差的绝对值不再 < 1,如果只是进行简单的左旋,得到的树仍然是不平衡的。我们应该按照如下图所示进行二次旋转:

  右左情况的右左旋转实例:

插入代码:

    /**
* 插入结点 (测试整形插入)
*/
public AVLNode<T> insertNode(AVLNode T,T value){
if(T==null){
T = new AVLNode(value);
}else{
//走左边
if((Integer)value<(Integer)T.data){
T.lchild = insertNode(T.lchild,value);
//分情况旋转 //判断是否需要旋转
if(getHeight(T.lchild)-getHeight(T.rchild)>=2){
//左左情况
if((Integer)value<(Integer)T.lchild.data){ //这是针对插入会出现的情况判断(也可以通过比较孩子结点的当前高度来判断)
// if(getHeight(T.lchild)>getHeight(T.rchild)){
//单旋-右旋
T = singleRotateWithRight(T);
}else{
//左右情况 (排除掉相同的元素,相同元素不允许再次插入) //先左转,再右转 LR
T = doubleRotateWithLeft(T);
}
} //走右边
}else if((Integer)value>(Integer)T.data){
T.rchild = insertNode(T.rchild,value); //分情况旋转
if(getHeight(T.rchild)-getHeight(T.lchild)>=2){
//右右情况
if((Integer)value>(Integer)T.rchild.data){ //这是针对插入会出现的情况判断(也可以通过比较孩子结点的当前高度来判断)
// if(getHeight(T.rchild)>getHeight(T.lchild)){
//单旋-左旋
T = singleRotateWithLeft(T);
}else{
//右左情况
T = doubleRotateWithRight(T);
}
}
}else{
//相同,不再进行插入
}
} //每次插入都要 计算高度,这个高度是递归式的!
T.height = Max(getHeight(T.lchild),getHeight(T.rchild))+1; //每一层递归结束之前要重新计算一下高度(因为可能插入了新结点)
return T;
}

AVL树的删除操作

分析:我们用插入的例子来分析删除操作

  1. 首先,如果要删除的结点比当前根节点大,那么就会走左边进去

  2. 走左边进去删除后,判断是否需要平衡的依据一定是:

    if(getHeight(T.rchild)-getHeight(T.lchild)>=2) //删掉左子树的孩子结点,肯定是右边可能会高过左边的情况
  3. 然后如果需要重新平衡(修复),那么要进行哪种修复呢

    观察图:知道如果删除的是左边的结点,那么会出现:LL或者RL这样的旋转

    RL:如果删除了左边结点的,出现了根节点2 的平衡因子从-1 变到了-2,那么再判断:根的右子树5的

    左子树和右子树的高度差,当T.rchild.lchild.height >T.rchild.rchild.height,即满足上面需要RL旋转的的情况

                    if(getHeight(T.rchild)-getHeight(T.lchild)>=2){
    //删掉左子树的孩子结点,肯定是右边可能会高过左边的情况
    if(getHeight(T.rchild.lchild)>getHeight(T.rchild.rchild)){
    //左子树的 右子树比左子树的左子树要高
    //RL旋转
    T = doubleRotateWithRight(T);
    //记住,因为旋转后,根节点会发生变化,一定要重新接收根结点
    }else{
    //RR
    T = singleRotateWithLeft(T);
    }
    }

    所有代码:

        /**
    * 删除操作
    */
    public AVLNode<T> deleteNode(AVLNode T,T value){
    if(T==null){
    return null;
    }else{
    //往左走
    if((Integer)value<(Integer) T.data){
    T.lchild = deleteNode(T.lchild,value); //函数返回新的根节点(所以要重新建立孩子与双亲间的联系)
    if(getHeight(T.rchild)-getHeight(T.lchild)>=2){ //删掉左子树的孩子结点,肯定是右边可能会高过左边的情况
    if(getHeight(T.rchild.lchild)>getHeight(T.rchild.rchild)){ //左子树的 右子树比左子树的左子树要高
    //RL旋转
    T = doubleRotateWithRight(T); //记住,因为旋转后,根节点会发生变化,一定要重新接收根结点
    }else{
    //RR
    T = singleRotateWithLeft(T);
    }
    }
    //往右走
    }else if((Integer)value>(Integer)T.data){
    T.rchild = deleteNode(T.rchild,value);
    if(getHeight(T.lchild)-getHeight(T.rchild)>=2){ //删掉右子树的孩子结点,肯定是左边可能会高过右边的情况
    if(getHeight(T.lchild.rchild)>getHeight(T.lchild.lchild)){
    //LR旋转
    T = doubleRotateWithLeft(T);
    }else{
    //LL
    T = singleRotateWithRight(T);
    }
    }
    }else{
    //找到了要删除的结点 //1. 没有左右孩子,删除的是叶子节点 (不用判断是否需要修复--旋转)
    if(T.lchild==null&&T.rchild==null){
    T = null;
    //2. 删除的结点只有左孩子或右孩子
    }else {
    if(T.lchild!=null){
    T = T.lchild;
    }else if(T.rchild!=null){
    T = T.rchild;
    }else{
    //3. 删除的结点左右孩子都有
    T.data = find_min_value(T.rchild); //找到最小节点,替换
    T.rchild = deleteNode(T.rchild,(T)T.data); //删除替换的最小的那个结点 //判断旋转
    if(getHeight(T.lchild)-getHeight(T.rchild)>=2){
    if(getHeight(T.lchild.rchild)-getHeight(T.lchild.lchild)>=2){
    //LR
    T = doubleRotateWithLeft(T);
    }else{
    //LL
    T = singleRotateWithRight(T);
    }
    }
    }
    }
    }
    }
    if(T!=null){
    //重新计算高度
    T.height = Max(getHeight(T.lchild),getHeight(T.rchild))+1;
    } //返回新的根节点
    return T; } /**
    * 找到最小的结点值
    */
    public T find_min_value(AVLNode T){ if(T.lchild==null){
    return (T) T.data;
    }else{
    return find_min_value(T.lchild);
    }
    } /**
    * 用于比较两棵子树高度,比较哪边高 ,用于节点高度 = Max(T.lchild.height,T.rchild.height)+1
    */
    public int Max(int lHeight,int rHeight){
    if(lHeight>=rHeight){
    return lHeight;
    }else{
    return rHeight;
    }
    } /**
    * 获取结点高度,因为可能计算高度的时候,左右孩子结点很可能为空,如果不用这个方法判断的话,会导致nullPointerException
    */
    public int getHeight(AVLNode T){
    if(T==null){
    return -1;
    }else{
    return T.height;
    }
    }

测试代码

        System.out.println();
System.out.println("测试AVL:");
//测试AVL
AVLTree<Integer> avlTree = new AVLTree<>(); avlTree.root = avlTree.insertNode(avlTree.root,1);
avlTree.root = avlTree.insertNode(avlTree.root,2);
avlTree.root = avlTree.insertNode(avlTree.root,3);
avlTree.root = avlTree.insertNode(avlTree.root,4);
avlTree.root = avlTree.insertNode(avlTree.root,5);
avlTree.root = avlTree.insertNode(avlTree.root,6);
avlTree.root = avlTree.insertNode(avlTree.root,7);
avlTree.root = avlTree.insertNode(avlTree.root,8);
avlTree.root = avlTree.insertNode(avlTree.root,10);
avlTree.root = avlTree.insertNode(avlTree.root,11);
avlTree.root = avlTree.insertNode(avlTree.root,12);
avlTree.root = avlTree.insertNode(avlTree.root,13);
avlTree.root = avlTree.insertNode(avlTree.root,14);
avlTree.root = avlTree.insertNode(avlTree.root,15); avlTree.levelTraverse(); System.out.println(); System.out.println("删除5,6,7");
avlTree.deleteNode(avlTree.root,5);
avlTree.deleteNode(avlTree.root,7);
avlTree.deleteNode(avlTree.root,6); avlTree.levelTraverse()

原来的为:

层序遍历测试结果:

通过debug查看结果:

删除5,6,7后:

二叉平衡树AVL的插入与删除(java实现)的更多相关文章

  1. Algorithms: 二叉平衡树(AVL)

    二叉平衡树(AVL):   这个数据结构我在三月份学数据结构结构的时候遇到过.但当时没调通.也就没写下来.前几天要用的时候给调好了!详细AVL是什么,我就不介绍了,维基百科都有.  后面两月又要忙了. ...

  2. (4) 二叉平衡树, AVL树

    1.为什么要有平衡二叉树? 上一节我们讲了一般的二叉查找树, 其期望深度为O(log2n), 其各操作的时间复杂度O(log2n)同时也是由此决定的.但是在某些情况下(如在插入的序列是有序的时候), ...

  3. 树-二叉平衡树AVL

    基本概念 AVL树:树中任何节点的两个子树的高度最大差别为1. AVL树的查找.插入和删除在平均和最坏情况下都是O(logn). AVL实现 AVL树的节点包括的几个组成对象: (01) key -- ...

  4. java项目---用java实现二叉平衡树(AVL树)并打印结果(详)(3星)

    package Demo; public class AVLtree { private Node root; //首先定义根节点 private static class Node{ //定义Nod ...

  5. AVL树(二叉平衡树)详解与实现

    AVL树概念 前面学习二叉查找树和二叉树的各种遍历,但是其查找效率不稳定(斜树),而二叉平衡树的用途更多.查找相比稳定很多.(欢迎关注数据结构专栏) AVL树是带有平衡条件的二叉查找树.这个平衡条件必 ...

  6. 从零开始学算法---二叉平衡树(AVL树)

    先来了解一些基本概念: 1)什么是二叉平衡树? 之前我们了解过二叉查找树,我们说通常来讲, 对于一棵有n个节点的二叉查找树,查询一个节点的时间复杂度为log以2为底的N的对数. 通常来讲是这样的, 但 ...

  7. 看动画学算法之:平衡二叉搜索树AVL Tree

    目录 简介 AVL的特性 AVL的构建 AVL的搜索 AVL的插入 AVL的删除 简介 平衡二叉搜索树是一种特殊的二叉搜索树.为什么会有平衡二叉搜索树呢? 考虑一下二叉搜索树的特殊情况,如果一个二叉搜 ...

  8. 高度平衡的二叉搜索树(AVL树)

    AVL树的基本概念 AVL树是一种高度平衡的(height balanced)二叉搜索树:对每一个结点x,x的左子树与右子树的高度差(平衡因子)至多为1. 有人也许要问:为什么要有AVL树呢?它有什么 ...

  9. 树-二叉搜索树-AVL树

    树-二叉搜索树-AVL树 树 树的基本概念 节点的度:节点的儿子数 树的度:Max{节点的度} 节点的高度:节点到各叶节点的最大路径长度 树的高度:根节点的高度 节点的深度(层数):根节点到该节点的路 ...

随机推荐

  1. Word2Vec小心得

    今天终于想明白了分层softmax的作用: 哈夫曼树的作用是什么??用平均最小的长度编码!编码是为了解码成信息! 神经概率语言模型:有映射层,隐藏层,输出层,假设隐藏层是300维,输出层是和单词的数量 ...

  2. python生成二维码图片

    依赖Pillow库. import qrcode # 简单用法 img = qrcode.make('Hello World!') img.save('qr_code.png') # 高级用法 qr ...

  3. 【Winfrom-DataTable填充ListView】ListView与DataTable相互转换 .net

    1.DataTable转成ListView: 先遍历DataTable的列,把DataTable列名添加到listView列头. 然后外循环添加行,内循环添加列 跟这篇文章是一样的 http://ww ...

  4. 【leetcode】1250. Check If It Is a Good Array

    题目如下: Given an array nums of positive integers. Your task is to select some subset of nums, multiply ...

  5. linux中more命令如何使用

         more命令,功能类似 cat ,cat命令是整个文件的内容从上到下显示在屏幕上.兄弟连Linux培训教程() more会以一页一页的显示方便使用者逐页阅读,而最基本的指令就是按空白键(sp ...

  6. UVA 10491 Cows and Cars (全概率公式)

    #include<bits/stdc++.h> #include<stdio.h> #include<iostream> #include<cmath> ...

  7. codevs 1405 牛的旅行x

    牛的旅行 [问题描述] 农民John的农场里有很多牧区.有的路径连接一些特定的牧区.一片所有连通的牧区称为一个牧场.但是就目前而言,你能看到至少有两个牧场不连通.现在,John想在农场里添加一条路径 ...

  8. 11.Python变量及其使用

    无论使用什么语言编程,其最终目的都是对数据进行处理.程序在编程过程中,为了处理数据更加方便,通常会将其存储在变量中. 形象地看,变量就像一个个小容器,用于“盛装”程序中的数据.除了变量,还有常量,它也 ...

  9. C++入门经典-例4.9-输出不同生命周期的变量值

    1:代码如下: // 4.9.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #include<iostream> using ...

  10. NP-Hard问题和NP-Complete问题

    对 NP-Hard问题和NP-Complete问题的一个直观的理解就是指那些很难(很可能是不可能)找到多项式时间算法的问题.因此一般初学算法的人都会问这样一个问题:NP-Hard和NP-Complet ...