【C++】平衡二叉树(AVL树)的实现

目录

  • 一、AVL树的概念
  • 二、AVL树的实现
    • 1、AVL树的定义
    • 2. 平衡二叉树的插入
      • 2.1 按照二叉排序树的方式插入并更新平衡因子
      • 2.2 AVL树的旋转
        • 2.2.1 新节点插入较高左子树的左侧(LL平衡旋转)
        • 2.2.2 新节点插入较高右子树的右侧(RR平衡旋转)
        • 2.2.3 新节点插入较高左子树的右侧(LR平衡旋转)
        • 2.2.4 新节点插入较高右子树的左侧(RL平衡旋转)
        • 2.2.5 总结
    • 3 平衡二叉树的删除(了解即可)
    • 4 平衡二叉树的验证
  • 三、平衡二叉树的效率分析

一、AVL树的概念

二叉排序树虽可以缩短查找的效率,但如果数据有序或接近有序二叉搜索树将退化为单支树,查找元素相当于在顺序表中搜索元素,效率低下。
为了避免树的高度增长过快,降低二叉排序树的性能,规定在插入和删除结点时,要保证任意结点的左、右子树高度差的绝对值不超过1,将这样的二叉树称为平衡二叉树,也称AVL树。

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

  • 它的左右子树都是AVL树
  • 左右子树高度之差(简称平衡因子)的绝对值不超过1(-1/0/1)在这里插入图片描述

二、AVL树的实现

1、AVL树的定义

AVL树结点的定义:

template<class K, class V>
struct AVLTreeNode
{AVLTreeNode(const pair<K, V>& kv):_left(nullptr),_right(nullptr),_parent(nullptr),_kv(kv),_bf(0){}AVLTreeNode<K, V>* _left;AVLTreeNode<K, V>* _right;AVLTreeNode<K, V>* _parent;		// 使用三叉链方便后续处理,但要记得维护pair<K, V> _kv;					// 保存键值对int _bf;						// 平衡因子
};

2. 平衡二叉树的插入

2.1 按照二叉排序树的方式插入并更新平衡因子

AVL树就是在二叉排序树的基础上加上了平衡因子,因此AVL树也可以看成是二叉排序树。那么AVL树的插入过程可以分为两步:
(1) 按照二叉排序树的方法插入新结点
(2) 调整结点的平衡因子

bool Insert(const pair<K, V>& kv)
{// 先按照二叉排序树的方法进行结点插入if (_root == nullptr){_root = new Node(kv);return true;}Node* parent = nullptr;Node* cur = _root;while(cur){if (kv.first < cur->_kv.first){parent = cur;cur = cur->_left;}else if (kv.first > cur->_kv.first){parent = cur;cur = cur->_right;}else{return false;}}cur = new Node(kv);if (kv.first < parent->_kv.first){parent->_left = cur;}else{parent->_right = cur;}cur->_parent = parent;// 新结点插入后,AVL树的平衡性可能会遭到破坏,此时就需要更新平衡因子,并检测是否// 破坏了AVL树的平衡性while (parent){/*cur插入后,parent的平衡因子一定需要调整,在插入之前,parent的平衡因子分为三种情况:-1,0, 1, 分以下两种情况:1. 如果cur插入到parent的左侧,只需给parent的平衡因子-1即可2. 如果cur插入到parent的右侧,只需给parent的平衡因子+1即可*/if (parent->_left == cur){--parent->_bf;}else{++parent->_bf;}/*此时:parent的平衡因子可能有三种情况:0,正负1, 正负21. 如果parent的平衡因子为0,说明插入之前parent的平衡因子为正负1,插入后被调整成0,此时满足AVL树的性质,插入成功2. 如果parent的平衡因子为正负1,说明插入前parent的平衡因子一定为0,插入后被更新成正负1,此时以parent为根的树的高度增加,需要继续向上更新3. 如果parent的平衡因子为正负2,则parent的平衡因子违反平衡树的性质,需要对其进行旋转处理*/if (0 == parent->_bf){break;}else if (1 == parent->_bf || -1 == parent->_bf){cur = cur->_parent;parent = parent->_parent;}else if (2 == parent->_bf || -2 == parent->_bf){// 旋转处理}else{// 如果平衡因子不是以上几种情况,说明代码逻辑错误assert(false);}}return true;
}

2.2 AVL树的旋转

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

2.2.1 新节点插入较高左子树的左侧(LL平衡旋转)

在这里插入图片描述
上图在插入前,AVL树是平衡的,新节点插入到30的左子树(注意:此处不是左孩子)中,30左子树增加了一层,导致以60为根的二叉树不平衡,要让60平衡,只能将60左子树的高度减少一层,右子树增加一层,即将左子树往上提,这样60转下来,因为60比30大,只能将其放在30的右子树,而如果30有右子树,右子树根的值一定大于30,小于60,只能将其放在60的左子树,旋转完成后,更新节点的平衡因子即可。

在旋转过程中,有以下几种情况需要考虑:

  1. 30节点的右孩子可能存在,也可能不存在
  2. 60可能是根节点,也可能是子树
    如果是根节点,旋转完成后,要更新根节点
    如果是子树,可能是某个节点的左子树,也可能是右子树
void RotateR(Node* parent)
{// subL:parent的左孩子// subLR:parent的左孩子的右孩子,注意:该点可能不存在Node* subL = parent->_left;Node* subLR = subL->_right;subL->_right = parent;parent->_left = subLR;Node* ppnode = parent->_parent;		// 记录parent的父结点,用于连接新的子树parent->_parent = subL;if (subLR){subLR->_parent = parent;}if (ppnode == nullptr){_root = subL;_root->_parent = nullptr;}else {if (ppnode->_left == parent){ppnode->_left = subL;}else{ppnode->_right = subL;}subL->_parent = ppnode;}// 根据调整后的结构更新部分节点的平衡因子subL->_bf = parent->_bf = 0;
}
2.2.2 新节点插入较高右子树的右侧(RR平衡旋转)

在这里插入图片描述
具体实现参考右旋即可。

void RotateL(Node* parent)
{Node* subR = parent->_right;Node* subRL = subR->_left;subR->_left = parent;parent->_right = subRL;Node* ppnode = parent->_parent;		// 记录parent的父结点parent->_parent = subR;if (subRL){subRL->_parent = parent;}if (ppnode == nullptr){_root = subR;_root->_parent = nullptr;}else{if (ppnode->_left == parent){ppnode->_left = subR;}else{ppnode->_right = subR;}subR->_parent = ppnode;}parent->_bf = subR->_bf = 0;
}
2.2.3 新节点插入较高左子树的右侧(LR平衡旋转)

在这里插入图片描述
将双旋变成单旋后再旋转,即:先对30进行左单旋,然后再对90进行右单旋,旋转完成后再考虑平衡因子的更新。

void RotateLR(Node* parent)
{// subL:parent的左孩子// subLR:parent的左孩子的右孩子,注意:该点可能不存在Node* subL = parent->_left;Node* subLR = subL->_right;// 旋转之前,保存pSubLR的平衡因子,旋转完成之后,需要根据该平衡因子来调整其他节点的平衡因子int bf = subLR->_bf;RotateL(parent->_left);RotateR(parent);if (1 == bf){subL->_bf = -1;}else if (-1 == bf){parent->_bf = 1;}
}
2.2.4 新节点插入较高右子树的左侧(RL平衡旋转)

在这里插入图片描述
参考右左双旋。

void RotateRL(Node* parent)
{Node* subR = parent->_right;Node* subRL = subR->_left;int bf = subRL->_bf;RotateR(parent->_right);RotateL(parent);if (1 == bf){parent->_bf = -1;}else if (-1 == bf){subR->_bf = 1;}
}
2.2.5 总结

假如以parent为根的子树不平衡,即parent的平衡因子为2或者-2,分以下情况考虑:

  1. parent的平衡因子为2,说明parent的右子树高,设parent的右子树的根为subR
    当subR的平衡因子为1时,执行左单旋
    当subR的平衡因子为-1时,执行右左双旋
  2. parent的平衡因子为-2,说明parent的左子树高,设parent的左子树的根为subL
    当subL的平衡因子为-1是,执行右单旋
    当subL的平衡因子为1时,执行左右双旋

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

3 平衡二叉树的删除(了解即可)

因为AVL树也是二叉排序树,可按照二叉排序树的方式将节点删除,然后再更新平衡因子,只不错与删除不同的时,删除节点后的平衡因子更新,最差情况下一直要调整到根节点的位置。
平衡二叉树删除操作的具体步骤:

  1. 先按照二叉排序树的方式删除结点
  2. 一路向上找到最小不平衡子树,找不到就结束
  3. 找最小不平衡子树下,最高的儿子和孙子
  4. 根据孙子的位置,调整平衡
    • 孙子在LL:右单旋
    • 孙子在RR:左单旋
    • 孙子在LR:先左旋再右旋
    • 孙子再RL:先右旋再左旋
  5. 如果不平衡向上传导,继续第二步
    • 对最小不平衡子树的旋转可能导致树变矮,从而导致上层祖先不平衡

4 平衡二叉树的验证

AVL树是在二叉搜索树的基础上加入了平衡性的限制,因此要验证AVL树,可以分两步:

  1. 验证其为二叉搜索树
    如果中序遍历可得到一个有序的序列,就说明为二叉搜索树
  2. 验证其为平衡树
    • 每个节点子树高度差的绝对值不超过1(注意节点中如果没有平衡因子)
    • 节点的平衡因子是否计算正确
// 求二叉树的高度
int _Height(Node* root)
{if (root == nullptr){return 0;}int leftH = _Height(root->_left);int rightH = _Height(root->_right);return leftH > rightH ? leftH + 1 : rightH + 1;
}
// 验证平衡树
bool _Isbalance(Node* root)
{if (root == nullptr){return true;}int leftH = _Height(root->_left);int rightH = _Height(root->_right);if (rightH - leftH != root->_bf){cout << root->_kv.first << "结点平衡因子异常" << endl;return false;}return rightH - leftH < 2&& _Isbalance(root->_left)&& _Isbalance(root->_right);
}

三、平衡二叉树的效率分析

在平衡二叉树上进行查找的过程与二叉排序树相同。因此,在查找过程中,进行关键字的比较次数不超过树的深度。假设以 n h n_h nh表示深度为h的平衡二叉树中含有的最少结点数。 n 0 = 0 , n 1 = 1 , n 2 = 2 n_0=0,n_1=1,n_2=2 n0=0,n1=1,n2=2,并且有 n h = n h − 2 + n h − 1 + 1 n_h=n_{h-2}+n_{h-1}+1 nh=nh2+nh1+1含有n个结点的平衡二叉树的最大深度为 O ( l o g 2 n ) O(log_2n) O(log2n),因此平均查找效率为 O ( l o g 2 n ) O(log_2n) O(log2n)
但是如果要对AVL树做一些结构修改的操作,性能非常低下,比如:插入时要维护其绝对平衡,旋转的次数比较多,更差的是在删除时,有可能一直要让旋转持续到根的位置。因此:如果需要一种查询高效且有序的数据结构,而且数据的个数为静态的(即不会改变),可以考虑AVL树,但一个结构经常修改,就不太适合。

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

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

相关文章

leetcode 二分查找·系统掌握 猜数字大小

题目&#xff1a; 题解&#xff1a; 使用最经典普通二分即可 int guessNumber(int n) {long l0,rn,mid;while(l<r){mid(rl)>>1;if(guess(mid)0)return mid;else if(guess(mid)-1)rmid-1;else lmid1;}return 0;}

全流程FVCOM水环境、污染物迁移、水交换、水质、潮流、温盐、波浪及泥沙数值模拟

近年来&#xff0c;随着计算技术的发展和对海洋、水环境问题认识的加深&#xff0c;数值模拟技术在海洋、水环境等科学研究中的应用越来越广泛。FVCOM因其独特的优点&#xff0c;成为研究海洋动力过程、污染物扩散、水质变化等问题的重要工具。作为一种基于有限体积法的数值模型…

第2章 Android应用的界面编程

&#x1f308;个人主页&#xff1a;小新_- &#x1f388;个人座右铭&#xff1a;“成功者不是从不失败的人&#xff0c;而是从不放弃的人&#xff01;”&#x1f388; &#x1f381;欢迎各位→点赞&#x1f44d; 收藏⭐️ 留言&#x1f4dd; &#x1f3c6;所属专栏&#xff1…

媒体邀约有啥要注意的

传媒如春雨&#xff0c;润物细无声&#xff0c;大家好&#xff0c;我是51媒体网胡老师。 媒体宣传加速季&#xff0c;100万补贴享不停&#xff0c;一手媒体资源&#xff0c;全国100城线下落地执行。详情请联系胡老师。 媒体邀约是邀请媒体参与活动或报道的重要过程&#xff0c…

搜索引擎推广基本概念与方法分享-华媒舍

销量是每个企业及个人在商业领域中追求的目标之一。而引擎霸屏推广就是一种高效的手段&#xff0c;通过该方法可以助你实现销量的狂揽。本文将为你科普引擎霸屏推广的基本概念与方法&#xff0c;帮助你了解如何运用这一有效的推广策略。 一、引擎霸屏推广 引擎霸屏推广指的是在…

【中学教资科目二】02中学课程

02中学课程 第一节 课程概述1.1 课程的分类 第二节 课程组织2.1 课程内容的文本表现形式2.2 课程评价 第三节 基础教育课程改革3.1 基础教育改革的目标3.2 新课改的课程结构 第一节 课程概述 1.1 课程的分类 学校课程有多种类型&#xff0c;其中最利于学生系统掌握人类所取得的…

软件缺陷及JIRA工具

一、软件缺陷及跟踪流程 1&#xff0c;软件缺陷信息 案例 &#xff08;1&#xff09;缺陷报告的基本内容 缺陷的标题 预置条件 重现步骤 期望结果 实际结果 &#xff08;2&#xff09;软件缺陷的状态 新建 打开 修复 关闭 &#xff08;3&#xff09;软件缺陷的严重程度 …

NeRF从入门到放弃2:InstantNGP

原始的NeRF每条光线上的点都要经过MLP的查询&#xff0c;才能得到其密度和颜色值&#xff0c;要查询的点非常多&#xff0c;而MLP的推理是比较耗时的。 InstantNGP将空间划分成多个层级的体素&#xff08;voxels&#xff09;&#xff0c;并且在每个体素内部使用神经网络来预测…

深入理解前端缓存

前端缓存是所有前端程序员在成长历程中必须要面临的问题&#xff0c;它会让我们的项目得到非常大的优化提升&#xff0c;同样也会带来一些其它方面的困扰。大部分前端程序员也了解一些缓存相关的知识&#xff0c;比如&#xff1a;强缓存、协商缓存、cookie等&#xff0c;但是我…

CTFHUB-SSRF-POST请求

通过file协议访问flag.php文件内容 ?urlfile:///var/www/html/flag.php 右键查看页面源代码 需要从内部携带key发送post数据包即可获得flag ?urlhttp://127.0.0.1/flag.php 得到了key 构造POST请求数据包&#xff0c;进行url编码&#xff08;新建一个txt文件&#xff0c;…

电力电源需要挑战的8大常见问题

电力电源需要挑战的8大常见问题 引言 如果说发电厂输出的是“干净”高质量的电源,但经过输配电,受天气、用户设备、人为因素损坏影响,电压过冲、跌落、中断、共模噪声等电源质量问题就相当突出。尤其在工业环境中,电源质量一般更差,不能忽视的是早年设计的大楼配电,已不能…

无线麦克风哪个品牌音质最好,揭秘k歌哪种麦克风音质好

无线领夹麦克风在自媒体行业中的地位日益凸显&#xff0c;它已经成为许多视频博主提升作品音频质量的秘密武器。不论是刚入门的自媒体新人&#xff0c;还是已经积累了一定粉丝基础的资深博主&#xff0c;都开始重视起音频设备的选择。一个好的领夹麦克风不仅能够提供清晰的录音…

经验总结--开关MOS管发热的一般原因/电源开发经验总结

开关MOS管发热的一般原因 做电源设计,或者做驱动方面的电路,难免要用到场效应管,也就是人们常说的MOS管。MOS管有很多种类,也有很多作用。做电源或者驱动的使用,当然就是用它的开关作用。 无论N型或者P型MOS管,其工作原理本质是一样的。MOS管是由加在输入端栅极的电压来控…

vben admin BasicTable表格基本使用

vben admin是一款强大的后台管理系统&#xff0c;广泛应用于各种项目中。本文将为您详细介绍如何使用 便您更快地上手并充分发挥其功能。 Table 表格 | Vben Admin一个开箱即用的前端框架https://jeesite.com/front/vben-admin/docs/components/table.html#usage 1.register:…

VirtualBox虚拟机下安装Ubuntu24.04操作系统

目录 0 背景1 虚拟机的安装1.1 下载安装包1.2 走安装向导 2 操作系统的安装2.1 下载光盘镜像文件2.2 安装操作系统到虚拟机上 3 基本配置3.1 网络连接方式3.2 共享文件夹3.3 设置显存大小 0 背景 首先说说Ubuntu系统&#xff0c;或者更普遍一点&#xff0c;Linux系统究竟有什么…

基于java+springboot+vue实现的智慧生活商城系统(文末源码+Lw)244

摘 要 计算机网络发展到现在已经好几十年了&#xff0c;在理论上面已经有了很丰富的基础&#xff0c;并且在现实生活中也到处都在使用&#xff0c;可以说&#xff0c;经过几十年的发展&#xff0c;互联网技术已经把地域信息的隔阂给消除了&#xff0c;让整个世界都可以即时通…

ionic7 从安装 到 项目启动最后打包成 apk

报错处理 在打包的时候遇到过几个问题&#xff0c;这里记录下来两个 Visual Studio Code运行ionic build出错显示ionic : 无法加载文件 ionic 项目通过 android studio 打开报错 capacitor.settings.gradle 文件不存在 说明 由于之前使用的是 ionic 3&#xff0c;当时打包的…

驾考小技巧:老北京布鞋!距离高考出分还剩3天,我却看到有些孩子已经拿了“满分”——早读(逆天打工人爬取热门微信文章解读)

我20年驾校4000多块钱&#xff0c;你呢&#xff1f; 引言Python 代码第一篇 洞见 距离高考出分还剩3天&#xff0c;我却看到有些孩子已经拿了“满分”第二篇 视频新闻结尾 引言 昨天的文章顺利发出 看来“梅西” 这两个字在我们这边 不是敏感词 只是很多个罗粉搞得有点过头了 …

Android设置页面Activity全屏(隐藏导航栏、状态栏)

3、代码中设置&#xff1a;在setContentView 之前调用 requestWindowFeature(Window.FEATURE_NO_TITLE); getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN); 注意&#xff1a; 当有全面屏手机可以显示虚拟…

公司名称含有关商标名称可能涉及侵权!

有个朋友找到普推商标知产老杨&#xff0c;说有个人给他打电话&#xff0c;说他的公司名称侵权另一家的商标名称&#xff0c;让他要改下公司名称&#xff0c;不改就要告侵权&#xff0c;此前看到过许多&#xff0c;在一些省市注册公司时&#xff0c;如果公司名称与已注册商标相…