当前位置: 首页 > news >正文

数据结构和算法之树形结构(3)

文章出处:数据结构和算法之树形结构(3)

关注码农爱刷题,看更多技术文章!!

四、平衡二叉树(接前篇)

      上一章节讲到为了避免二叉查找树退化成链表后的极度不平衡带来的低效率而衍生出了平衡二叉树,平衡二叉树的严格定义是这样的:二叉树中任意一个节点的左右子树的高度相差不能大于 1。这样的定义就是尽可能保证二叉树左右节点的平衡,从而保证时间复杂度的高效。从平衡二叉树的定义不难看出,不仅满二叉树,还有完全二叉树(关于两者的定义,可以查看前述文章 数据结构和算法之线性结构)都是符合平衡二叉树定义的,而其他的非完全二叉树也可能是平衡二叉树,例如下图就是一棵非完全二叉树,但它是平衡二叉树。

图片

     了解平衡二叉树基本定义后,我们再看平衡二叉查找树,所谓平衡二叉查找树就是不仅要满足平衡二叉树的定义,还需要满足二叉查找树的规范即按大小排序。常见的平衡二叉查找树有AVL树红黑,本章先介绍AVL树。

五、AVL树

     AVL树是最早设计出来的一种平衡二叉查找树,它通过在插入和删除节点时进行旋转操作来保持内部树的平衡,即任何节点的左右子树高度相差不超过1,是一种严格的自平衡二叉树。这种即时的高度自平衡机制是AVL树最大的特点,也使得其在执行查找、插入和删除操作时的效率非常高,其时间复杂度保持在O(log n)。

     AVL树自平衡机制

     要了解AVL树的自平衡机制,我们要先了解一些基本概念:

平衡因子:即AVL树每个节点都有一个平衡因子,定义为其左子树的高度减去右子树的高度;平衡因子的值只能是-1、0或1。

     通俗的说,平衡因子就是AVL树节点的左右子树高度的差的负值,如果左子树比右子树高一层,那么平衡因子就为-1;如果左右子树一样高,平衡因子就为0;如果右子树比左子树高一层,那么平衡因子就为1,这三种情况下AVL树的性质都没有被打破。按照这个规则,如果平衡因子为-1到1以外的其他值,则说明左右子树已经失衡,AVL树性质被打破,AVL树则不再是一棵AVL树。而对AVL树进行增删节点,就会破坏这种平衡即失衡,所以AVL树引进了一种机制,在增删节点后,通过节点自旋来恢复被破坏的平衡性,这种机制即AVL树的自平衡机制。所谓节点自旋,就是在不改变AVL树节点大小顺序的情况下调整树的结构使之重新平衡(这个平衡本质上就是其左右子树高度的平衡)。

      对于AVL树失衡的四种场景和对应的旋转方式总结如下表:

图片

      LL场景-右旋

     下面我们以LL场景为例,对AVL树失衡和通过旋转实现自平衡的过程进行说明。所谓LL失衡,即节点左子树高度大于右子树高度且高差大于1,并且其左子节点的左子树也高于或等于右子树的高度但差在平衡因子值正常范围内,如下图:

图片

      图中根节点8左子树高度-右子树高度 = 2,即失衡;而其左子节点6左子树高度也高于右子树高度但高度差为1。调整的旋转方法则向相反方向旋转,即右旋,具体步骤如下:

1.  右旋失衡根节点8使其成为其左节点6的右节点;2.  同时使原左节点6的右孩子7成为原根节点8的左节点

       旋转后的图如下,旋转后,恢复了AVL树的平衡特性:

图片

   左旋过程类似,右旋和左旋实现代码如下:


private AVLNode rightRotate(AVLNode red) {AVLNode yellow = red.left;AVLNode green = yellow.right;yellow.right = red;red.left = green;return yellow;
}private AVLNode leftRotate(AVLNode red) {AVLNode yellow = red.right;AVLNode green = yellow.left;yellow.left = red;red.right = green;return yellow;
}
     LR场景-左右旋

    下面我们再用稍微复杂的RL场景为例,对AVL树失衡和通过旋转实现自平衡的过程进行进一步说明。所谓LR失衡,即失衡节点左子树高度大于右子树高度且高差大于1,并且失衡节点左子节点的右孙子树高于左孙子树的高度但差在平衡因子值正常范围内,如下图:

图片

     图中根节点6左子树高度-右子树高度 = 2,即失衡;而其左子节点6的右孙子树高于左孙子树且高度差为1。调整的旋转方法则采用右旋,具体步骤如下:

1. 先左旋左节点2,使其右孩子4成为其父节点;   同时右孩子4成为根节点6的左节点,操作完后如下图:

图片

2. 再右旋根节点6,使其成为左节点4的右节点;   同时,左节点4成为新的根节点,而节点5则成为原根节点6的左节点

    左右旋转都完成后图如下:

图片

    其实,LR场景就是RR场景和LL场景的组合,第一步左旋后,LR场景就从RR场景转变为了LL场景,所以后续的处理也对应LL场景处理方法右旋即可,这点从代码实现上也可以看出:


private AVLNode leftRightRotate(AVLNode root) {root.left = leftRotate(root.left); // 先左旋左节点,对应RRreturn rightRotate(root); // 再右旋根节点 对应LL
}private AVLNode rightLeftRotate(AVLNode root) {root.right = rightRotate(root.right);return leftRotate(root);
}

     RL场景的右左旋,逻辑则刚好相反,大家可自行去研究,AVL树的自平衡机制和相关知识就介绍到此(其节点增删查找同二叉查找树一致,此处也不再重复介绍,可参看前述文章)。正由于AVL树的自平衡机制,使AVL树能持续保持平衡二叉树的特性,从而能保证其时间复杂度一直接近O(logN),且处理旋转的时间复杂度为O(1),因而整体上会比非平衡二叉查找树有更好的效率。

      但是成也萧何、败也萧何,AVL树的优点在某些场景下也会变成缺点,例如在大规模节点频繁插入和删除的场景下,实现自平衡机制的旋转操作就会成为负担,所以,红黑树应运而生。红黑树相关知识,请关注后续章节。

码农爱刷题

为计算机编程爱好者和从业人士提供技术总结和分享 !为前行者蓄力,为后来者探路!


http://www.mrgr.cn/news/34904.html

相关文章:

  • 在ubuntu上安装ubuntu22.04并ros2 humble版本的docker容器记录
  • 【ES6】ES6中,如何实现桥接模式?
  • 校园二手交易网站毕业设计基于SpringBootSSM框架
  • PostgreSQL序列:创建、管理与高效应用指南
  • 操作系统——进程调度
  • 国标GB28181视频平台EasyCVR私有化部署视频平台对接监控录像机NVR时,录像机“资源不足”是什么原因?
  • 花半小时用豆包Marscode 和 Supabase免费部署了一个远程工作的导航站
  • 2025 广州国际新能源汽车功率半导体技术展览会与您相约广州
  • linux文件目录指令合集--拷贝、移动、查看
  • 到底是谁配谁-《分析模式》漫谈33
  • 【附实例】Python字典的各种操作
  • c++哈希
  • 算法:二维数组查找
  • UWB为什么是首选的室内定位技术
  • 【VMware】虚拟机安装
  • 基于Java+Jsp+SpringMVC漫威手办商城系统设计和实现
  • 蓝牙技术|详谈蓝牙信道探测技术,可实现厘米级精准定位
  • Google 提供基于AI的模糊测试框架
  • Axure-本地发布,局域网内用户访问
  • 如何使用MacPorts安装tesseract来进行简单的OCR识别
  • C++中stack类和queue类
  • 【Canvas与诗词】铁马冰河入梦来
  • lambda表达式详解
  • AI赋能千人千面营销:从数据采集到精准用户画像的全流程解析
  • 亚马逊云科技的成功秘诀:韧性与持续创新的经验之道
  • .NET 一款新的内网对抗综合利用工具