高阶DS---AVL树详解(每步配图)

目录

前言:

AVL树的概念:

AVL树节点的定义:

AVL树的插入(重点)

AVL树的旋转:

(1)新节点插入较高左子树的左侧---右单旋

(2)新节点插入较高右子树的右侧---左单旋

(3)新节点插入较高左子树的右侧---左右双旋

(4)新节点插入较高右子树的左侧---右左双旋

总结:

AVL树的验证:

验证用例:

AVL树的删除(了解):

AVL树性能分析:

结语:


前言:

如果有友友需要本文章的全部源码的话请前往AVL树源码

AVL树的概念:

二叉搜索树虽可以缩短查找的效率,但如果数据有序或接近有序二叉搜索树将退化为单支树,查找元素相当于在顺序表中搜索元素,效率低下。因此,两位俄罗斯的数学家G.M.Adelson-Velskii和E.M.Landis在1962年发明了一种解决上述问题的方法:当向二叉搜索树中插入新结点后,如果能保证每个结点的左右子树高度之差的绝对值不超过 1(需要对树中的结点进行调整),即可降低树的高度,从而减少平均搜索长度。

一棵AVL树或者是空树,或者是具有以下性质的二叉搜索树:

(1)它的左右子树都是AVL树

(2)左右子树高度之差(简称平衡因子)的绝对值不超过1(-1/0/1)

如果一棵二叉搜索树是高度平衡的,它就是AVL树。如果它有n个结点,其高度可保持在O(logN) ,搜索时间复杂度O(logN)。

例如下图就是一个AVL树圆圈外面的数字就是平衡因子,为右子树高度 - 左子树高度。

AVL树节点的定义:

为了AVL树实现简单,AVL树节点在定义时维护一个平衡因子和采用孩子双亲表示法,具体节点定义如下:

因为在实际开发时我们都是树节点单独创建一个类不是很经常使用静态内部类,故我们这里就创建一个TreeNode类来实现。

public class TreeNode {public int val;//节点值public int bf;//AVL树的平衡因子public TreeNode left;//左孩子引用public TreeNode right;//右孩子引用public TreeNode parent;//父亲节点引用public TreeNode(int val){this.val = val;}}

注意:

当前节点的平衡因子 = 右子树高度 - 左子树的高度。但是,不是每棵树,都必须有平衡因子,这只是其中的一种实现方式。

AVL树的插入(重点)

AVL树就是在二叉搜索树的基础上引入了平衡因子,因此AVL树也可以看成是二叉搜索树。那么AVL树的插入过程可以分为两步:

(1)按照二叉搜索树的方式插入新节点。

(2)调整节点的平衡因子。

但是在插入节点后要更新平衡因子,这时AVL树的平衡性就可能会遭到破坏,我们就要进行调整。

假设插入节点为node,node的父亲节点为parent,在node节点插入后,parent节点的平衡因子一定要进行调整,在插入之前,parent的平衡因子分为三种情况:-1,0,1。

(1)如果cur插入到parent的左侧,只需给parent的平衡因子 -1 即可。

(2)如果cur插入到parent的右侧,只需给parent的平衡因子 +1 即可。

此时:parent的平衡因子可能有三种情况:0,正负1, 正负2。对应分析如下:

(1)如果parent的平衡因子为0,说明插入之前parent的平衡因子为正负1,插入后被调整成0,此时满足AVL树的性质,插入成功,就不需要向上调整,因为对于上面的节点来说这颗子树的最大高度没有改变。

(2)如果parent的平衡因子为正负1,说明插入前parent的平衡因子一定为0,插入后被更新成正负1,此时以parent为根的树的高度增加,需要继续向上更新。

(3)如果parent的平衡因子为正负2,则parent的平衡因子违反平衡树的性质,需要对其进行旋转处理。

根据上面的分析我们可以先写出如下代码(大体框架),首先我们先根据二叉搜索树的查找节点方式找到要插入节点的父亲节点,插入节点后修改平衡因子,根据修改后平衡因子的情况分为三种情况。👍👍👍

public boolean insert(int val){//根据二叉搜索树查找节点的方式,找到插入点TreeNode node = new TreeNode(val);//根节点为空if(root == null){root = node;}TreeNode cur = root;TreeNode parent = null;//parent始终是cur的父亲节点,当cur为null时parent就是我们要插入节点的父亲节点while(cur != null){if(cur.val > val){//去左子树找parent = cur;cur = cur.left;}else if(cur.val < val){//去右子树找parent = cur;cur = cur.right;}else{//插入节点已存在,插入失败return false;}}//插入节点if(parent.val < val){parent.right = node;}else{parent.left = node;}node.parent = parent;cur = node;//调整插入节点父亲节点的平衡因子while(parent != null){if(cur == parent.left){parent.bf--;}else{parent.bf++;}//当调整后父亲节点平衡因子为0if(parent.bf == 0){//说明插入后,parent树的左右最大深度不变//已经平衡了break;}else if(parent.bf == 1 || parent.bf == -1){//说明插入后,parent树的左右最大深度改变,会影响parent树的parent的平衡因子//要继续向上修改平衡因子cur = parent;parent = cur.parent;}else{//parent的平衡因子为2,要进行旋转调整//有两种情况分别为右树高和左树高if(parent.bf == 2){if(cur.bf == 1){//左旋rotateLeft(parent);}else{//进行右左双旋//其实就是先调整成能左旋的情况再左旋//cur.bf == -1rotateRL(parent);}}else{//parent.bf == -2左树高if(cur.bf == -1){//右旋rotateRight(parent);}else{//左右双旋//先左旋成能将整体进行右旋的情况再进行右旋rotateLR(parent);//cur.bf == 1}}//break;}}return true;}

如果对上面代码的else部分为什么是这么旋转感到疑惑的话,可以先跳过,在介绍完下面AVL树的旋转后就明白了。 

AVL树的旋转:

如果在一棵原本是平衡的AVL树中插入一个新节点,可能造成不平衡,此时必须调整树的结构,使之平衡化。根据节点插入位置的不同,AVL树的旋转分为四种:

注意:下面我画的四张图大家一定一定要会画,总不可能记代码吧,没有意义。

我最常用的例子节点是:60,30,a,b,c(其中a,b,c为满足要求的任意值)

(1)新节点插入较高左子树的左侧---右单旋

上面的图便是我们右单旋的全过程了,我们可以发现旋转完后这个AVL树变得平衡了,且需要修改的平衡因子只有两个subL和parent的平衡因子。

具体代码如下:

在这过程中subLR节点可能不存在故要来一次特判防止空指针异常,接着要分是根节点和不是根节点两种情况,如果不是根节点又要分是pParent(parent的父亲节点)的左子树还是右子树。主要要弄好指向和平衡因子的修改(经过分析和作图后发现调整后只有subL和parent平衡因子发生改变)。

private void rotateRight(TreeNode parent){//右旋TreeNode subL = parent.left;//parent节点的左孩子节点TreeNode subLR = subL.right;//parent节点的左孩子节点的右孩子节点parent.left = subLR;//防止节点不存在,空指针异常if(subLR != null){subLR.parent = parent;}subL.right = parent;//在修改parent的父亲节点时,要提前记入下来防止丢失TreeNode pParent = parent.parent;parent.parent = subL;//如果parent是根节点,即没有父亲节点if(parent == root){root = subL;subL.parent = null;}else{//有父亲节点故要考虑其是父亲节点的左孩子还是右孩子if(pParent.left == parent){pParent.left = subL;}else{pParent.right = subL;}subL.parent = pParent;}//经过分析和作图发现调整后只有subL和parent平衡因子发生改变//subL平衡因子从-1变成0//parent平衡因子从-2变成0subL.bf = 0;parent.bf = 0;}

(2)新节点插入较高右子树的右侧---左单旋

上图为我们左单旋的全过程,这个其实可以仿照我们右单旋的步骤来。具体代码如下:

用private进行封装,创建对应的孩子节点,进行下面个个参数的指向修改时一定要画图🌸🌸🌸,节点的选取可以仿照我上面画的,最后注意不要忘了修改对应节点的平衡因子。

这里教给大家一个记忆小技巧:对哪个节点进行旋转,新的parent节点的旋转方向(根据名字)节点要断掉。

private void rotateLeft(TreeNode parent){//左旋//小技巧:对哪个节点进行旋转,新的parent节点的旋转方向(根据名字)节点要断掉TreeNode subR = parent.right;//parent的右孩子TreeNode subRL = subR.left;//parent的右孩子的左孩子//防止节点不存在空指针异常if(subRL != null){subRL.parent = parent;}//这里建议画图理解parent.right = subRL;subR.left = parent;//在修改parent的父亲节点时,要提前记入下来防止丢失TreeNode pParent = parent.parent;parent.parent = subR;if(parent == root){root = subR;subR.parent = null;}else{if(pParent.left == parent){pParent.left = subR;}else{pParent.right = subR;}subR.parent = pParent;}subR.bf = 0;parent.bf = 0;}

(3)新节点插入较高左子树的右侧---左右双旋

在有些情况下只进行左旋和右旋还并不能解决所有情况,例如下图,如果友友感兴趣的话可以自己试试,显然一次旋转完成不了。我们正确的旋转方式为左右双旋,先左旋再进行右旋。

正确旋转过程如下图。

下图只演示了subLR的平衡因子为-1的情况还有1和0的情况就交给友友们自己去完成了都差不多的👍👍👍 

代码如下:

这里特别注意我们传入左右旋的方法的参数是传入parent而不是其孩子节点,这个一定要弄清楚否则就错了,下面之所以没有bf == 0的情况是因为在 bf == 0 的情况下在rotateLeft方法和rotateRight方法下就已经吧要修改的bf修改完成了。但是不能if后面用else必须是else if,因为else会把bf == 0的情况收纳进去这样就出错了。 

private void rotateLR(TreeNode parent){//左右双旋TreeNode subL = parent.left;TreeNode subLR = subL.right;int bf = subLR.bf;//bf的获取必须在旋转之前否则会因旋转而改变,旋转会改变对应的平衡因子rotateLeft(subL);rotateRight(parent);//画图,这里可以分为插在左边还是右边if(bf == -1){parent.bf = 1;subL.bf = 0;subLR.bf = 0;}else if(bf == 1){//bf == 1parent.bf = 0;subLR.bf = 0;subL.bf = -1;}}

(4)新节点插入较高右子树的左侧---右左双旋

右左双旋的实现,友友们可以参考左右双旋。

具体流程和左右双旋差不多也有三种情况,bf为0,1,-1的三种情况,注意传入左右旋方法的参数是传入对应的parent节点。

对应代码如下:

private void rotateRL(TreeNode parent){//右左双旋TreeNode subR = parent.right;TreeNode subRL = subR.left;int bf = subRL.bf;//bf的获取必须在旋转之前否则会因旋转而改变//旋转传入的父亲节点为原来的不是改变后的rotateRight(subR);rotateLeft(parent);//画图,这里可以分为插在左边还是右边if(bf == 1){parent.bf = -1;subRL.bf = 0;subR.bf = 0;}else if(bf == -1){// bf == -1subRL.bf = 0;parent.bf = 0;subR.bf = 1;}}

总结:

新节点插入后,假设以pParent为根的子树不平衡,即pParent的平衡因子为2或者-2,分以下情况考虑:

1.pParent的平衡因子为2,说明pParent的右子树高,设pParent的右子树的根为pSubR。

(1)当pSubR的平衡因子为1时,执行左单旋。

(2)当pSubR的平衡因子为-1时,执行右左双旋。

2.pParent的平衡因子为-2,说明pParent的左子树高,设pParent的左子树的根为pSubL。

(1)当pSubL的平衡因子为-1是,执行右单旋。

(2)当pSubL的平衡因子为1时,执行左右双旋。

即:pParent与其较高子树节点的平衡因子时同号时单旋转,异号时双旋转。

旋转完成后,原pParent为根的子树个高度降低,已经平衡,不需要再向上更新。

AVL树的验证:

分为两步:

(1)验证其为二叉搜索树

如果中序遍历可得到一个有序的序列,就说明为二叉搜索树

(2)验证其为平衡树

1.每个节点子树高度差的绝对值不超过1(注意节点中如果没有平衡因子)

2.节点的平衡因子是否计算正确

对应代码如下:

public boolean isBalance(TreeNode root){if(root == null) return true;int heightL = height(root.left);int heightR = height(root.right);if(heightR - heightL != root.bf){System.out.println(root.val + ":的平衡因子计算错误");return false;}return Math.abs(heightL - heightR) <= 1 && isBalance(root.left) && isBalance(root.right);}private int height(TreeNode root){if(root == null) return 0;int heightL = height(root.left);int heightR = height(root.right);return Math.max(heightL,heightR) + 1;}public void inOrder(TreeNode root){if(root == null) return;inOrder(root.left);System.out.print(root.val + " ");inOrder(root.right);}

验证用例:

大家可以自己完成AVL树的代码后把下面这三个实例带进去验证,如果是true且中序遍历为升序的话代码就没什么问题了。

这里再补充一个用例:int[] array = {30,20,90,60,180,40};

AVL树的删除(了解):

因为AVL树也是二叉搜索树,可按照二叉搜索树的方式将节点删除,然后再更新平衡因子,只不过与删除不同的是,删除节点后的平衡因子更新,最差情况下一直要调整到根节点的位置。

这里由于实现比较麻烦要考虑的东西很多(且面试一般也不会让你写代码)文章篇幅有限只说大致流程:

1、找到需要删除的节点。

2、按照搜索树的删除规则删除节点。

3、更新平衡因子,如果出现了不平衡,进行旋转。单旋,双旋。

AVL树性能分析:

AVL树是一棵绝对平衡的二叉搜索树,其要求每个节点的左右子树高度差的绝对值都不超过1,这样可以保证查询时高效的时间复杂度,即 。但是如果要对AVL树做一些结构修改的操作,性能非常低下,比如:插入时要维护其绝对平衡,旋转的次数比较多,更差的是在删除时,有可能一直要让旋转持续到根的位置。因此:如果需要一种查询高效且有序的数据结构,而且数据的个数为静态的(即不会改变),可以考虑AVL树,但一个结构经常修改,就不太适合。

结语:

其实写博客不仅仅是为了教大家,同时这也有利于我巩固知识点,和做一个学习的总结,由于作者水平有限,对文章有任何问题还请指出,非常感谢。如果大家有所收获的话还请不要吝啬你们的点赞收藏和关注,这可以激励我写出更加优秀的文章。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/785027.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

unity双层滑动实现

实现功能&#xff1a; 当滑动列表中内容处于顶端的时候&#xff0c;向上滑动优先滑动整个滑动列表&#xff0c;当滑动列表移动到设置位置&#xff0c;即设定的最高处时&#xff0c;继续移动列表内内容。向下移动亦然&#xff0c;当内容处于滑动列表顶端时&#xff0c;移动整个滑…

深入解析Hadoop生态核心组件:HDFS、MapReduce和YARN

这里写目录标题 01HDFS02Yarn03Hive04HBase1&#xff0e;特点2&#xff0e;存储 05Spark及Spark Streaming关于作者&#xff1a;推荐理由&#xff1a;作者直播推荐&#xff1a; 一篇讲明白 Hadoop 生态的三大部件 进入大数据阶段就意味着进入NoSQL阶段&#xff0c;更多的是面向…

[游戏开发][UE5.3]代码生成蓝图文件并在代码中保存文件。

我看网上有人的做法比我更好&#xff0c;我这个更简单 UE5-GAS:读取Excel数据在蓝图创建并更新GE类 - 知乎 数据配表 测试编辑器API 创建编辑器蓝图文件&#xff0c;继承AssetActionUtility.h 创建在编辑器中显示的函数&#xff0c;可以用中文命名方便其他人使用。 右键任意…

从零开始学Python数据分析:菜鸟也能成高手(文末送书)

&#x1f935;‍♂️ 个人主页&#xff1a;艾派森的个人主页 ✍&#x1f3fb;作者简介&#xff1a;Python学习者 &#x1f40b; 希望大家多多支持&#xff0c;我们一起进步&#xff01;&#x1f604; 如果文章对你有帮助的话&#xff0c; 欢迎评论 &#x1f4ac;点赞&#x1f4…

【数据结构与算法初阶(c语言)】插入排序、希尔排序、选择排序、堆排序、冒泡排序、快速排序、归并排序、计数排序-全梳理(万字详解,干货满满,建议三连收藏)

目录 1.排序的概念及其运用 1.1排序的概念 1.2排序运用 1.3常见的排序算法 2.插入排序 2.1 原理演示&#xff1a;​编辑 2.2 算法实现 2.3 算法的时间复杂度和空间复杂度分析 3.希尔排序 3.1算法思想 3.2原理演示 3.3代码实现 3.4希尔算法的时间复杂度 4.冒泡排序 4.1冒泡排…

test01

欢迎关注博主 Mindtechnist 或加入【Linux C/C/Python社区】一起学习和分享Linux、C、C、Python、Matlab&#xff0c;机器人运动控制、多机器人协作&#xff0c;智能优化算法&#xff0c;滤波估计、多传感器信息融合&#xff0c;机器学习&#xff0c;人工智能等相关领域的知识和…

Java的IDEA的工程管理

模块和包的图标&#xff1a; 举个例子&#xff1a; IDEA中创建包&#xff1a; 如图所示&#xff0c;com.LBJ的意思是在com包中创建子包LBJ 参见&#xff1a; IDEA中项目、模块和包的关系_idea中模块和项目-CSDN博客

网站秒收录的6个方法

网站怎么做到秒收录 网站的收录速度对于网站的曝光和流量至关重要。如果能够实现网站的秒收录&#xff0c;将大大提高网站的可见性和访问量。下面介绍几种方法&#xff0c;帮助您实现网站秒收录的目标。 1. 提交sitemap.xml文件 创建并提交sitemap.xml文件是实现网站快速收录…

Netty学习——源码篇9 Netty的Handler其他处理 备份

1 ChannelHandlerContext 每个ChannelHandler被添加到ChannelPipeline后&#xff0c;都会创建一个ChannelHandlerContext&#xff0c;并与ChannelHandler关联绑定。ChannelHandlerContext允许ChannelHandler与其他的ChannelHandler进行交互。ChannelHandlerContext不会改变添加…

U8二次开发-钉钉集成

钉钉开放平台作为企业沟通和协作的重要工具,其技术的每一次迭代都为企业带来了新的机遇和挑战。随着企业对于高效沟通和智能化管理的需求日益增长,钉钉平台的SDK更新显得尤为重要。把传统的U8与钉钉平台集成,可以有效的将业务功能和角色进行前移,打破应用系统二八原则,即8…

dailyneaty、希亦、鲸立婴儿洗衣机怎么样?三款卷王测评PK对决

曾经我还是一直选择手洗婴儿衣物&#xff0c;最终还是加入了买婴儿洗衣机的大军&#xff0c;一方面因为我懒&#xff0c;不想再继续手洗&#xff0c;另一方面是因为我看了科普才知道&#xff0c;当我们清洗衣物时&#xff0c;除了要洗掉衣物表面的污渍&#xff0c;更需要消除掉…

MATLAB——知识点备忘

最近在攻略ADC建模相关方面&#xff0c;由好多零碎的知识点&#xff0c;这里写个备忘录。 Matlab 判断一个数是否为整数 1. isinteger 函数 MATLAB中&#xff0c;可以使用 isinteger 函数来判断一个数是否为整数&#xff0c;例如&#xff1a;要判断x是否为整数可以采用以下代…

科研学习|论文解读——情感对感知偶然信息遭遇的影响研究(JASIST,2022)

原文题目 Investigating the impact of emotions on perceiving serendipitous information encountering 一、引言 serendipity一词最初是由霍勒斯沃波尔创造的&#xff0c;他将其定义为“通过意外和睿智发现你并不追求的事物”。信息研究中大多数现有的偶然性定义从几个角度看…

劳动力规划:对企业加速运营的未来展望

近年来&#xff0c;企业面临着过山车般的经济形势&#xff0c;面对消费水平的上涨、市场波动带来的担忧以及数字化的加速转型&#xff0c;许多企业虽然对未来仍秉持着谨慎乐观的态度&#xff0c;但也同时认为自身缺乏持续增长和成功转型的能力。为了让企业能够实现战略目标、应…

算法笔记之蓝桥杯pat系统备考(3)

算法笔记之蓝桥杯&pat系统备考&#xff08;2&#xff09; 多训练、多思考、多总结٩(๑•̀ω•́๑)۶ 八、深搜和广搜 8.1DFS dfs是一种枚举完所有完整路径以遍历所有情况的搜索方法&#xff0c;可以理解为每次都是一条路走到黑的犟种。 以老朋友斐波那契额数列为例&a…

Docker-compose管理工具的使用

华子目录 容器编排工具docker composecompose介绍compose使用的三个步骤docker-compose.yml文件案例compose具有管理应用程序整个生命周期的命令 docker compose安装安装条件在Linux系统上安装composedocker compose卸载 docker compose运用演示修改compose配置&#xff0c;添加…

【手册】——mq延迟队列

目录 一、背景介绍二、思路&方案三、过程1.项目为啥用延迟队列&#xff1f;2.项目为啥用三方延迟队列&#xff1f;3.项目中为啥用rabbitmq延迟队列&#xff1f;4.rabbitmq延迟队列的安装5.rabbitmq的延迟队列配置方式5.1.exchange配置5.2.queues配置5.3.exchange和queues的…

初识C++ · 入门(2)

目录 1 引用 1.1引用的概念 1.2 引用的特性 2 传值&#xff0c;传引用的效率 3 引用和指针的区别 4 内联函数 4.1 内联函数的定义 4. 2 内联函数的特性 5 关键字auto 5.1关于命名的思考 5.2 关于auto的发展 5.3 auto使用规则 6 范围for的使用 7 空指针 1 引用 …

win10如何开启麦克风权限,win10麦克风权限设置

手机下载软件后,总是会跳出各种权限需要,例如访问通讯录、读取位置信息、启动相机等等。电脑上的应用也有这些权限设置,比如说玩游戏、直播、或录制视频时,我们需要打开麦克风权限,否则无法进行交流和录音。但是,win10如何开启麦克风权限呢?针对这个问题,小编已整理了两…

《自动机理论、语言和计算导论》阅读笔记:p115-p138

《自动机理论、语言和计算导论》学习第 6 天&#xff0c;p115-p138 总结&#xff0c;总计 24 页。 一、技术总结 1.associativity and comutativity (1)commutativity(交换性): Commutativity is the property of an operator that says we can switch the order of its ope…