【搜索结构】AVL树的学习与实现

目录

什么是AVL树

AVL树的定义

插入函数的实现

左单旋和右单旋

左右双旋与右左双旋


什么是AVL树

        AVL树实际上就是二叉搜索树的一种变体,我们都知道二i叉搜索树可以将查找的时间复杂度提升到O(logn),极大提升搜索效率。但是在极端情况下,当按顺序向树插入节点时,二叉树严重不平衡,相当于退化成了链表,此时查找的时间复杂度就变为了O(n),这并不是我们希望看到的。

        那么有没有什么方式可以让二叉搜索树保持一定的平衡性从而不至于导致查找效率严重降低呢?AVL树也就是高度平衡二叉树给出的解决方案是:

1. 二叉树的每个节点都有一个平衡因子,平衡因子等于左子树高度减右子树高度的值

2. 平衡因子的绝对值不能超过1

3. 当插入或删除节点导致平衡因子绝对值超过1时,进行旋转

AVL树的定义

        让我们来思考一下,要实现前面描述的功能,AVL树的单个节点应该有哪些成员变量呢?

1. 首先肯定要有左右子树的节点

2. 然后为了旋转时能够找到父亲,我们还需要存父亲节点

3. 为了确保平衡,我们要将左右高度差作为平衡因子保存

4. 最后还有搜索要用到的键值对

AVL树节点类定义:

template<class K, class V>
struct AVLTreeNode
{// 由于插入节点时要向上更新,所以使用三叉链结构AVLTreeNode<K, V>* left;AVLTreeNode<K, V>* right;AVLTreeNode<K, V>* parent; pair<K, V> _kv; // 键值int _bf; // 平衡因子AVLTreeNode(const pair<K, V>& kv):_kv(kv), left(nullptr),right(nullptr),parent(nullptr),_bf(0){}
};

那么,AVL树的定义就应该是:

template<class K, class V>
class AVLTree
{typedef AVLTreeNode<K, V> Node;
private:Node* _root = nullptr;
};

插入函数的实现

        让我们先明确一下AVL树插入一个节点要做的事:

1. 按照二叉搜索树的规则找到插入位置进行插入

2. 根据左右高度差得到平衡因子

3. 当平衡因子绝对值大于1时进行旋转处理

        首先二叉树搜索规则就是:当插入节点的键小于当前节点的键时,和当前节点的左子树进行比较;当插入节点的键大于当前节点的键时,和当前节点的右子树进行比较;否则说明插入节点的键已经存在,这不符合二叉搜索树的规则,直接报错。我们按照这个规则找到可以插入的位置然后将新节点插入。

        插入完成之后,我们开始更新平衡因子,当新节点位于父亲的左边时,bf减1;当位于父亲的右边时,bf加1。修改完父亲的平衡因子后,进行判断:

1. 如果当前父亲平衡因子值为0,说明高度差没有改变,不需要进行处理,直接break即可。

2. 如果当前父亲平衡因子值为1/-1,说明插入导致高度差改变了,这可能会导致祖先节点的平衡因子绝对值超过1,所以需要继续往上更新祖先的平衡因子

3. 如果更新后的祖先的平衡因子绝对值超过1,就需要进行旋转处理

为了更直观地理解这个过程,让我们来看一个简单的例子:

插入新节点导致祖先失去平衡:

通过旋转,使AVL树恢复平衡

我们可以看到,这棵树的根节点平衡因子在插入新节点后,由1变为2,而进行一次左旋之后,平衡因子更新为0,恢复了平衡。

由于旋转涉及到的情况比较多且有一些细节操作,而这只是其中最简单的一种情况,所以我们先写出AVL树插入的基本框架,之后再对各个需要旋转的情况分别进行处理。

	bool Insert(const pair<K, V>& kv){if (_root == nullptr){_root = new Node(kv);return true;}Node* parent = nullptr; // 用于记录插入位置的父亲节点Node* cur = _root; // 用于比较查找到插入位置while (cur){parent = cur;if (kv.first < cur->_kv.first){cur = cur->left;}else if (kv.first > cur->_kv.first){cur = cur->right;}// 搜索二叉树不支持重复key值的情况else{return false;}}// 此时说明已经找到了插入位置,插入新节点cur = new Node(kv);if (kv.first < parent->_kv.first){parent->left = cur;}else{parent->right = cur;}cur->parent = parent; // 记得保持三叉链结构// 更新平衡因子while (parent){if (cur == parent->left){parent->_bf--;}else{parent->_bf++;}// 正好平衡了if (parent->_bf == 0){break;}// 说明需要向上调整else if (parent->_bf == 1 || parent->_bf == -1){cur = parent;parent = parent->parent;}// 在此处进行旋转调整else if (parent->_bf == 2 || parent->_bf == -2){// 旋转处理……}// 说明旋转有问题,不能在|bf|==2时恢复平衡else{assert(false);}}return true;}

左单旋和右单旋

        先来看左单旋,如下图所示,是左单旋最简单的情况:

但显然需要左单旋的情况通常会更复杂些,所以我们实现左单旋时,需要考虑具有通用性的情况:

        可以发现,我们的左单旋操作似乎只需要让parent的右指向subRL,然后让subR的左指向parent。但大家可别忘了,我们定义AVL树节点时,为了方便向上更新,设计的是三叉链结构,所以还需要更新subRL和parent的父亲节点。除此之外,还需要考虑到parent可能是祖先节点的孩子,所以如果parent是AVL树的根节点:将根节点更新为subR,并将subR的父亲更新为nullptr;如果parent是祖先节点的孩子:则将祖先节点的孩子更新为subR,并将subR的父亲更新为父亲节点。

        指针朝向修改完后,我们还需要修改发生高度变化的节点的平衡因子,如上图所示,parent的平衡因子由2变为0,subR的平衡因子由1变为0。

void RotateL(Node* parent)
{// 在修改之前先保存祖父节点Node* grandpa = parent->parent;// 事实上,对于左旋这一情况,我们要修改的只有parent,subR,subRL最多再加个grandpa的指针朝向Node* subR = parent->right;Node* subRL = subR->left;// 进行左旋操作parent->right = subRL;subR->left = parent;// 之后还得把父指针也一起修改了parent->parent = subR;if(subRL)subRL->parent = parent;// 这棵树不是子树if (parent == _root){_root = subR;_root->parent = nullptr;}// 这棵树是子树else{// 是祖父节点的左子树if (grandpa->left == parent){grandpa->left = subR;}// 是祖父节点的右子树else{grandpa->right = subR;}subR->parent = grandpa;}parent->_bf = subR->_bf = 0;}

        接下来是右单旋,在局部子树左偏时,我们通过右旋来进行处理:

由于在左单旋部分我们已经详细讲解过了,右单旋其实就相当于反过来,所以就不再讲解一遍了。

	void RotateR(Node* parent){// 在修改之前先保存祖父节点Node* grandpa = parent->parent;// 事实上,对于右旋这一情况,我们要修改的只有parent,subL,subLR最多再加个grandpa的指针朝向Node* subL = parent->left;Node* subLR = subL->right;// 进行右旋操作parent->left = subLR;subL->right = parent;// 之后还得把父指针也一起修改了parent->parent = subL;if (subLR)subLR->parent = parent;// 需要考虑现在调整的这棵树是子树的情况if (parent == _root){_root = subL;_root->parent = nullptr;}else{// 是祖父节点的左子树if (grandpa->left == parent){grandpa->left = subL;}// 是祖父节点的右子树else{grandpa->right = subL;}subL->parent = grandpa;}parent->_bf = subL->_bf = 0;
}

左右双旋与右左双旋

        我们还是先来看一个左右双旋的简单例子,可以看到,我们先通过一次左旋,把这棵子树修改为了单纯的左偏,而处理左偏,我们只需要进行一次右旋即可。

再来看更普遍的情况:

事实上,由于我们已经有了左旋和右旋的代码,所以进行双旋时,可以直接复用左旋和右旋函数,所以双旋主要考虑的是如何更新平衡因子。

一共有三种情况:

第一种:60就是新增节点,此时平衡因子全部更新为0

第二种:新增节点向b插入,此时parent的因子为1,其余因子为0

第三种:新增节点向c插入,此时subL的因子为-1,其余因子为0

大家可以自己分别画一下这三种情况,其实自己动手画一下就很好理解了,让我们看代码:

void RotateLR(Node* parent){Node* subL = parent->left;Node* subLR = subL->right;// 啊啊啊,原来是这里错了,调试了一个晚上。。// int bf = parent->_bf;int bf = subLR->_bf;RotateL(parent->left);RotateR(parent);if (bf == 0){parent->_bf = subL->_bf = subLR->_bf = 0;}else if (bf == -1){parent->_bf = 1;subLR->_bf = 0;subL->_bf = 0;}else if (bf == 1){parent->_bf = 0;subLR->_bf = 0;subL->_bf = -1;}//else//{//	assert(false);//}}

右左双旋和左右双旋完全类似,也是三种情况,其实理解了左右双旋,右左双旋就很好写了!

	void RotateRL(Node* parent){Node* subR = parent->right;Node* subRL = subR->left;// 提前记录修改点的平衡因子int bf = subRL->_bf;// 先右旋RotateR(parent->right);// 再左旋RotateL(parent);if (bf == 0){// 自己就是新增节点parent->_bf = subR->_bf = subRL->_bf = 0;}else if (bf == -1){// 新增节点位于subRL的左子树parent->_bf = 0;subRL->_bf = 0;subR->_bf = 1;}else if (bf == 1){// 新增节点位于subRL的右子树parent->_bf = -1;subRL->_bf = 0;subR->_bf = 0;}//else//{//	assert(false);//}}

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

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

【专题】2024年中国消费者消费意愿调查报告汇总PDF洞察(附原数据表)

原文链接&#xff1a;https://tecdat.cn/?p38242 当今时代&#xff0c;经济社会多元发展&#xff0c;消费市场复杂多变。消费者的行为、需求和支出意愿不断演变&#xff0c;深刻影响着各个领域的发展。家庭余钱的用途反映出消费者在储蓄、教育、医疗等方面的考量。在消费领域…

推荐一款游戏玩家性能优化工具:Razer Cortex

Razer Cortex是一款专为游戏玩家设计的性能优化工具&#xff0c;它旨在提升玩家的游戏体验。通过该软件&#xff0c;用户可以优化 PC 性能&#xff0c;从而提高游戏的流畅度&#xff0c;减少延迟并增强视觉效果&#xff0c;尤其在需要精准操作的游戏中&#xff0c;流畅的画面和…

人工智能(AI)对于电商行业的变革和意义

![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/402a907e12694df5a34f8f266385f3d2.png#pic_center> &#x1f393;作者简介&#xff1a;全栈领域优质创作者 &#x1f310;个人主页&#xff1a;百锦再新空间代码工作室 &#x1f4de;工作室&#xff1a;新空间代…

1435:【例题3】曲线 一本通 代替三分

1435&#xff1a;【例题3】曲线 题目来源&#xff1a;一本通oj链接 代替三分 题意 给出t组数据&#xff0c;每组里面有n个函数&#xff0c;求出t组数据的函数的最小值 思路 函数是二次函数&#xff0c;具有单峰性&#xff0c;利用左右两边单调性的原理可以进行答案三分处…

英伟达Isaac Manipulator产品体验

相关配置 Isaac Manipulator3.1.0Isaac Sim4.2.0Ubuntu20.04GPURTX 4090 LaptopCPUI9 13900HXMem64GB 过程记录与反馈 GPU加速效果 请描述您在使用Isaac Manipulator时&#xff0c;调用cuMotion加速库来进行机器人运动规划和轨迹优化等任务的步骤和过程&#xff0c;并记录任…

“非法”操控lambda(python)

能过python解释器关卡即是合法脚本代码&#xff0c;偶尔的“违规”操控也是一种唯美。 (笔记模板由python脚本于2024年11月13日 11:18:21创建&#xff0c;本篇笔记适合熟悉python的lambda操控的coder翻阅) 【学习的细节是欢悦的历程】 Python 官网&#xff1a;https://www.pyth…

[ 网络安全介绍 5 ] 为什么要学习网络安全?

&#x1f36c; 博主介绍 &#x1f468;‍&#x1f393; 博主介绍&#xff1a;大家好&#xff0c;我是 _PowerShell &#xff0c;很高兴认识大家~ ✨主攻领域&#xff1a;【渗透领域】【数据通信】 【通讯安全】 【web安全】【面试分析】 &#x1f389;点赞➕评论➕收藏 养成习…

java八股笔记-1-java基础

java 特点&#xff1a; 1.平台无关性&#xff0c;java 的字节码文件可以在任何安装了 JVM 的系统上运行 2.面相对象&#xff0c;几乎一切都可以抽象为对象&#xff0c;包括类&#xff0c;对象&#xff0c;继承&#xff0c;封装&#xff0c;多态&#xff0c;抽象 抽象&#xf…

Java入门16——接口

我们今天来学习接口&#xff0c;和继承有点像&#xff0c;话不多说&#xff0c;开始正题~ 一、接口 1.为什么要用接口 接口其实和继承很像&#xff0c;但是继承是 is-a 的关系&#xff0c;接口是 has-a 的关系&#xff0c;而且继承只能是一对一的关系&#xff0c;但是接口可以…

Sigrity SPEED2000 Power Ground Noise Simulation模式如何进行串扰分析操作指导-trace耦合

Sigrity SPEED2000 Power Ground Noise Simulation模式如何进行串扰分析操作指导-trace 耦合 Sigrity Power SI Power Ground Noise Simulation模式可以用来分析信号间的串扰,以下图为例 2D视图

地下水数值模拟软件Visual modflow Flex实践技术应用

专题一 地下水数值软件的操作流程、建模步骤和所需资料处理及相关注意事项 [1] Visual MODFLOW Flex特征 [2] Visual MODFLOW Flex软件界面及模块 [3] 地下水数值模拟的建模步骤及数据需求 专题二 模型建模操作方法 技巧、真实案例演练、特殊问题处理[1] 直接模型建模的操作方法…

保险、银行等金融行业都在做的“双录”是什么?电子签约如何实现

“双录”也就是同步录音、录像&#xff0c;是指在特定的业务场景中通过录音和录像的方式来记录相关业务过程中的关键环节和重要内容&#xff0c;帮助确定业务办理人真实身份和意愿、实现业务过程可回溯管理。 起初&#xff0c;双录主要用于保险销售&#xff0c;后来逐步扩展到…

总结拓展十五:特殊采购业务——寄售采购

1、寄售采购的定义 寄售采购是指供应商提供物料&#xff0c;并将它们存储在你处&#xff0c;在贵公司将这些物料从寄售库存提取&#xff08;转自有&#xff09;之前&#xff0c;该供应商一直是这些物料法律上的所有者。只有当这些物料被贵司转自有领用后&#xff0c;供应商才会…

python 同时控制多部手机

在这个智能时代,我们的手机早已成为生活和工作中不可或缺的工具。无论是管理多个社交媒体账号,还是处理多台设备上的事务,如何更高效地控制多个手机成为了每个人的痛点。 今天带来的这个的软件为你提供了一键控制多部手机的强大功能。无论是办公、娱乐,还是社交,你都能通过…

c++:string(一)

文章目录 一string类1C语言中的字符串2C中的string二遍历1[ ]2迭代器3const迭代器4范围for5auto6总结三String的尾插1size和length2max_size,capacity和clear3访问接口4尾插字符和字符串5 append的重载三string的扩容问题&#xff08;1&#xff09;怎么扩容&#xff08;2&#…

如何从数字化迈向智能化的跨越,重塑企业合同管理的未来

随着信息技术的快速发展&#xff0c;越来越多的企业开始认识到合同管理的重要性&#xff0c;并纷纷实施数字化战略以提高管理效率和降低运营成本。然而&#xff0c;仅仅实现合同管理的数字化还远远不够&#xff0c;真正的转型应该是向智能化迈进。本文将通过一个实际案例来探讨…

书生浦语XTuner 微调个人小助手

文章目录 一、环境配置与数据准备1.构建一个xtuner环境2.安装 XTuner3.修改提供的数据四、训练启动1.模型位置2.创建软连接即可3.修改官方的Config4.启动微调4.权重转换4. 模型合并二、进阶任务2.1 上传到 HuggingFace 一、环境配置与数据准备 XTuner 文档链接&#xff1a;XTu…

信捷 XDH PLC C语言 Ethercat 简易绝对运动 BMC_A_DRVA_BODY函数

本文以简易运动为例&#xff0c;描述多轴运动的程序封装。具有一定的参数价值。适用于信捷XDH PLC。 很容易移植到具有Ethercat 总线的PLC,使用ST语言的情况。 1.建立结构体 2.在全局变量表建立全局变量 &#xff08;1&#xff09;DRVA_PAR_array是类型为BMC_A_DRVA&#xff…

磐石云黑名单管理系统

黑名单验证平台是一款基于历史高风险号码实时验证的管理平台&#xff1b; 功能特点&#xff1b; 1、支持代理商账户 2、支持对接三方黑名单库进行缓存&#xff08;俗称扒库&#xff09;&#xff0c;首次获取黑名单后缓存到本地&#xff0c;下次不再付费调用三方接口&#xf…

Objects工具类详解

在 Java 编程中&#xff0c;对象的处理是不可避免的。为了简化对象操作并减少空指针异常&#xff08;NullPointerException&#xff09;的风险&#xff0c;Java 7 引入了 java.util.Objects 类。这个类包含了一系列静态方法&#xff0c;旨在帮助开发者更安全、更高效地处理对象…