AVL树

AVL树,也称平衡二叉搜索树,AVL是其发明者姓名简写。AVL树属于树的一种,而且它也是一棵二叉搜索树,不同的是他通过一定机制能保证二叉搜索树的平衡,平衡的二叉搜索树的查询效率更高。

AVL树特点

  • AVL树是一棵二叉搜索树。
  • AVL树的左右子节点也是AVL树。
  • AVL树拥有二叉搜索树的所有基本特点。
  • 每个节点的左右子节点的高度之差的绝对值最多为1,即平衡因子为范围为[-1,1]。

图中红色数字表示对应节点的高度,可以看到同一层的节点高度差都没有超过1。

二叉搜索树的平衡

基础的二叉搜索树构建出来可能会存在不平衡的现象,比如极端情况下,按照A B C D E F G H顺序插入树中,结果为,

但实际上我们更想要平衡一点的二叉搜索树,因为平衡的二叉搜索树能有效提高查询效率,比如上面的要查询“H”节点则需要比较8个节点才找到,而平衡的二叉搜索树只需要比较3个节点。

所以AVL树的出现就是为了解决平衡性问题,它的核心内容就是平衡处理机制,即所谓的旋转,一共有四种形式的旋转:右单旋、左单旋、左右双旋和右左双旋。

为什么要旋转

不管是什么方式的旋转,旋转的目的是为了降低树的高度,使其平衡,假如树结构如下图,

将“A”节点添加到树中,变成如下结构,树产生了不平衡,于是检查哪里不平衡,当到“C”节点时发现高度差超过1,

所以需要对“C”节点进行右单旋操作将高度降到2,达到平衡。

插入方式

AVL树一共有四种插入方式,根据插入方式不同需要做不同的旋转操作,现在往下看四种插入方式,设受插入节点影响而失去平衡的节点的父节点为Z,

  • LL插入方式,插入的节点在Z节点的左子树的左子树上,如下图,“A”节点插入影响“C”节点的平衡,“C”的父节点为“E”,插入节点“A”在“E”节点的左子树的左子树上。即“B”节点的左右子节点都算LL插入。

  • RR插入方式,插入的节点在Z节点的右子树的右子树上,如下图,“I”节点插入影响“G”节点的平衡,“G”的父节点为“E”,插入节点“I”在“E”节点的右子树的右子树上。即“H”节点的左右子节点都算RR插入。

  • LR插入方式,插入的节点在Z节点的左子树的右子树上,如下图,“C”节点插入影响“B”节点的平衡,“B”的父节点为“E”,插入节点“C”在“E”节点的左子树的右子树上。即“D”节点的左右子节点都算LR插入。

  • RL插入方式,插入的节点在Z节点的右子树的左子树上,如下图,“G”节点插入影响“H”节点的平衡,“H”的父节点为“E”,插入节点“G”在“E”节点的右子树的左子树上。即“F”节点的左右子节点都算RL插入。

右单旋

右单旋用于处理LL插入方式,假设存在一棵树,如下,

现插入“A”节点,假如不进行旋转的话,树结构为下图,所以遍历过程也会检查哪里不平衡,检查到“C”节点和“G”节点的高度差大于1,而且插入节点“A”属于“E”节点左子树的左子树,于是进行右单旋,

“C”节点右单旋即将“C”节点提高,原本它的父节点“E”则变为其右子节点,“C”节点原来的右子节点则变为其父节点“E”的左子节点。右单旋后的结果如下,重新达到了平衡。

左单旋

左单旋用于处理RR插入方式,假设存在一棵树,如下,

现插入“I”节点,假如不进行旋转的话,树结构为下图,所以遍历过程也会检查哪里不平衡,检查到“C”节点和“G”节点的高度差大于1,而且插入节点“I”属于“E”节点的右子树的右子树,于是进行左单旋,

“G”节点左单旋即将“G”节点提高,原本它的父节点“E”则变为其左子节点,“G”节点原来的左子节点则变为其父节点“E”的右子节点。左单旋后的结果如下,重新达到了平衡。

左右双旋

左右双旋用于处理LR插入方式,假设存在一棵树,如下,

现插入“C”节点,假如不进行旋转的话,树结构为下图,遍历过程会检查哪里不平衡,检查到“B”节点和“G”节点的高度差大于1,而且插入节点“C”属于“E”节点的左子树的右子树,于是进行左右双旋,

先以“D”节点为轴进行左单旋,结果为,

再以“D”节点为轴进行右单旋,得到最终结果,

右左双旋

右左双旋用于处理RL插入方式,假设存在一棵树,如下,

现插入“G”节点,假如不进行旋转的话,树结构为下图,遍历过程会检查哪里不平衡,检查到“C”节点和“H”节点的高度差大于1,而且插入节点“G”属于“E”节点的右子树的左子树,于是进行右左双旋,

先以“F”节点为轴进行右单旋,结果为,

再以“F”节点为轴进行左单旋,得到最终结果,

插入

空树时插入节点“E”直接作为根节点,“E”节点高度设为1,

继续插入“B”节点,小于“E”节点则添加到左边,且“E”节点高度加1,

继续插入“G”节点,大于“E”节点则添加到右边,此时“E”节点高度不变,

继续插入“D”节点,最终到“B”节点的右子节点,此时“B”节点高度加1,“E”节点高度也加1,

继续插入“C”节点,最终到“D”节点的左子节点,此时“D”、“B”、“E”节点高度都分别加1,并且先发现节点“D”与它同级节点(不存在即高度为0)高度差大于1,并且属于RL插入方式,使用右左双旋处理,

以“C”节点为轴进行右单旋,结果为,

再以“C”节点为轴进行左单旋,结果如下,可以看到进过右左双旋操作后二叉树已经达到平衡了。

总结,插入时可能会遇到四种不同的插入方式,分别是:LL插入方式、RR插入方式、LR和RL插入方式。根据不同的插入方式对应做旋转操作即能使树达到平衡状态。

查找

AVL树因为属于二叉搜索树,所以查找时与BST树完全一样,比如下面这棵树,查找“D”节点,

从根节点“C”开始,

“D”大于“C”,所以往右继续查找,

“D”小于“E”,所以往左查找,找到。

删除

删除操作主要分两种情况,一种是删除后不会影响平衡,那么直接按照BST树规则删除。另外一种是删除后会影响树的平衡,那么则需要再做旋转处理。

情况一

如树的结构,要删除“B”节点,

直接找到“B”节点,且因为是叶子节点,直接删掉即可。

最终为,

但如果删除的不是“B”节点,而是“C”节点,则不能直接删除“C”节点,

应该先找到“C”节点的前驱,它的前驱为“B”节点,使用“B”替换“C”节点,

最后将原来的“B”节点删除。

情况二

如树的结构,要删除“F”节点,

先找到“F”节点,

然后将“F”节点删除,此时导致了“C”节点和“G”节点的高度差超过1,需要做旋转操作,

而且因为C节点的左子节点高度比右子节点高度大,所以执行右单旋操作,旋转后为,

作者:超人汪小建
链接:https://juejin.im/post/5b6b897df265da0fab404318
来源:掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

06-看图理解数据结构与算法系列(AVL树)的更多相关文章

  1. 09-看图理解数据结构与算法系列(B树)

    B树 B树即平衡查找树,一般理解为平衡多路查找树,也称为B-树.B_树.是一种自平衡树状数据结构,能对存储的数据进行O(log n)的时间复杂度进行查找.插入和删除.B树一般较多用在存储系统上,比如数 ...

  2. 19-看图理解数据结构与算法系列(Radix树)

    Radix树 Radix树,即基数树,也称压缩前缀树,是一种提供key-value存储查找的数据结构.与Trie不同的是,它对Trie树进行了空间优化,只有一个子节点的中间节点将被压缩.同样的,Rad ...

  3. 13-看图理解数据结构与算法系列(Trie树)

    Trie树 Trie树,是一种搜索树,也称字典树或单词查找树,此外也称前缀树,因为某节点的后代存在共同的前缀.它的key都为字符串,能做到高效查询和插入,时间复杂度为O(k),k为字符串长度,缺点是如 ...

  4. 11-看图理解数据结构与算法系列(B树的删除)

    删除操作 删除操作比较复杂,主要是因为删除的项可能在叶子节点上也可能在非叶子节点上,而且删除后可能导致不符合B树的规定,这里暂且称之为导致B树不平衡,于是要进行一些合并.左旋.右旋等操作,使之符合B树 ...

  5. 10-看图理解数据结构与算法系列(B+树)

    B+树 B+树是B树的一种变体,也属于平衡多路查找树,大体结构与B树相同,包含根节点.内部节点和叶子节点.多用于数据库和操作系统的文件系统中,由于B+树内部节点不保存数据,所以能在内存中存放更多索引, ...

  6. 17-看图理解数据结构与算法系列(NoSQL存储-LSM树)

    关于LSM树 LSM树,即日志结构合并树(Log-Structured Merge-Tree).其实它并不属于一个具体的数据结构,它更多是一种数据结构的设计思想.大多NoSQL数据库核心思想都是基于L ...

  7. 看图轻松理解数据结构与算法系列(NoSQL存储-LSM树) - 全文

    <看图轻松理解数据结构和算法>,主要使用图片来描述常见的数据结构和算法,轻松阅读并理解掌握.本系列包括各种堆.各种队列.各种列表.各种树.各种图.各种排序等等几十篇的样子. 关于LSM树 ...

  8. Java数据结构和算法(七)--AVL树

    在上篇博客中,学习了二分搜索树:Java数据结构和算法(六)--二叉树,但是二分搜索树本身存在一个问题: 如果现在插入的数据为1,2,3,4,5,6,这样有序的数据,或者是逆序 这种情况下的二分搜索树 ...

  9. 数据结构与算法:AVL树

    AVL树 在计算机科学中,AVL树是最先发明的自平衡二叉查找树.在AVL树中任何节点的两个子树的高度最大差别为1,所以它也被称为高度平衡树.增加和删除可能需要通过一次或多次树旋转来重新平衡这个树.AV ...

随机推荐

  1. IEEE Trans 2006 使用K-SVD构造超完备字典以进行稀疏表示(稀疏分解)

    K-SVD可以看做K-means的一种泛化形式,K-means算法总每个信号量只能用一个原子来近似表示,而K-SVD中每个信号是用多个原子的线性组合来表示的.    K-SVD算法总体来说可以分成两步 ...

  2. That Nice Euler Circuit UVALive - 3263 || 欧拉公式

    欧拉定理: 简单多面体的顶点数V.棱数E及面数F间有关系有著名的欧拉公式:V-E+F=2. 设G为任意的连通的平面图,则v-e+f=2,v是G的顶点数,e是G的边数,f是G的面数.(引) 证明(?) ...

  3. Oracle查看所有表空间的数据使用情况

    -- 查看所有表空间的数据使用情况 SELECT Upper(F.TABLESPACE_NAME) "表空间名", D.TOT_GROOTTE_MB "表空间大小(M)& ...

  4. 移动端UI自动化Appium测试——Windows系统Appium环境配置

    1.安装JDK,官网下载即可,这里用的1.8,环境变量配置 2.安装Android sdk,API >= 17,环境变量配置 3.安装Nodejs,官网http://nodejs.org/dow ...

  5. 好用的SqlParamterList

    public class SqlParameterList : List<SqlParameter> { #region Properties /// <summary> // ...

  6. Java多线程——进程和线程

    Java多线程——进程和线程 摘要:本文主要解释在Java这门编程语言中,什么是进程,什么是线程,以及二者之间的关系. 部分内容来自以下博客: https://www.cnblogs.com/dolp ...

  7. 001原始编译全志r6平台tinav3.0.2系统

    001原始编译全志r6平台tinav3.0.2系统 2018/6/8 11:32 版本:V1.0 开发板:R6 SDK:tina v3.0.2 1.01原始编译全志r16平台tinav3.0系统: r ...

  8. SugarCRM安装踩雷(一)

    安装SugarCRM前置条件: 1.找对平台.正确版本的安装包 2.APACHE + MYSQL + TOMCAT环境先确保OK 坑1: 进入安装参数设置步骤的MYSQL用户密码——这里根据Mysql ...

  9. win7+idea+maven搭建spark源码阅读环境

    1.参考. 利用IDEA工具编译Spark源码(1.60~2.20) https://blog.csdn.net/He11o_Liu/article/details/78739699 Maven编译打 ...

  10. API设计指南(译)

    API的设计在软件系统中的重要性不言而喻,在swift.org上看到一篇“API Design Guidelines”,虽然是就Swift而言,但对于其它语言也有不少可以借鉴的地方,在这里粗略翻译一二 ...