【数据结构】平衡二叉树(AVL树)

目录

前言

一、AVL树概念

二、AVL树节点定义

 三、AVL树插入

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

2. 维护节点的平衡因子与调整树的结构

 a. 新节点插入较高左子树的左侧---左左:右单旋

b. 新节点插入较高右子树的右侧---右右:左单旋

c. 新节点插入较高左子树的右侧---左右:先左单旋再右单旋 

d. 新节点插入较高右子树的左侧---右左:先右单旋再左单旋

四、AVL树删除

附录:AVL树实现参考代码 


前言

二叉搜索树虽可以缩短查找的效率,但如果数据有序或接近有序二叉搜索树将退化为单支树,查 找元素相当于在顺序表中搜索元素,效率低下。

因此,两位俄罗斯的数学家G.M.Adelson-Velskii 和E.M.Landis在1962年发明了一种方法,用以解决上述问题。


一、AVL树概念

AVL 树是一种平衡二叉树,得名于其发明者的名字( Adelson-Velskii 以及 Landis)。(可见名字长的好处,命名都能多占一个字母出来)。平衡二叉树递归定义如下:

  • 左右子树的高度差小于等于 1。
  • 其每一个子树均为平衡二叉树。

 为了使AVL树保持平衡,在节点中增加了一个平衡因子作为节点属性。当树发生变化时,就通过维护平衡因子与改变树的结构来使树保持平衡。

二、AVL树节点定义

template<class K, class V>
struct AVLTreeNode
{AVLTreeNode<K, V>* _left;AVLTreeNode<K, V>* _right;AVLTreeNode<K, V>* _parent;std::pair<K, V> _kv;int _bf; //balance factorAVLTreeNode(const std::pair<K, V>& kv):_left(nullptr), _right(nullptr), _parent(nullptr), _kv(kv), _bf(0){}
};

 三、AVL树插入

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

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

参考二叉搜索树。

【数据结构】二叉搜索树-CSDN博客icon-default.png?t=N7T8https://blog.csdn.net/lyhv_v/article/details/139243506

2. 维护节点的平衡因子与调整树的结构

 cur插入后,pParent的平衡因子一定需要调整,在插入之前,pParent 的平衡因子分为三种情况:-1,0, 1, 分以下两种情况:

  1. 如果pCur插入到pParent的左侧,只需给pParent的平衡因子-1即可
  2. 如果pCur插入到pParent的右侧,只需给pParent的平衡因子+1即可

此时:pParent的平衡因子可能有三种情况:0,正负1, 正负2

  1. 如果pParent的平衡因子为0,说明插入之前pParent的平衡因子为正负1,插入后被调整成0,此时满足 AVL树的性质,插入成功
  2. 如果pParent的平衡因子为正负1,说明插入前pParent的平衡因子一定为0,插入后被更新成正负1,此 时以pParent为根的树的高度增加,需要继续向上更新
  3. 如果pParent的平衡因子为正负2,则pParent的平衡因子违反平衡树的性质,需要对其进行旋转处理

 a. 新节点插入较高左子树的左侧---左左:右单旋

Node* RotateR(Node* parent)
{Node* sub = parent->_left;Node* subR = sub->_right;if (parent == _root){_root = sub;sub->_parent = nullptr;}else{Node* up = parent->_parent;if (up->_kv.first > sub->_kv.first)up->_left = sub;elseup->_right = sub;sub->_parent = up;}parent->_left = subR;if (subR != nullptr)subR->_parent = parent;sub->_right = parent;parent->_parent = sub;parent->_bf = 0;sub->_bf = 0;return sub;
}

b. 新节点插入较高右子树的右侧---右右:左单旋

Node* RotateL(Node* parent)
{Node* sub = parent->_right;Node* subL = sub->_left;if (parent == _root){_root = sub;sub->_parent = nullptr;				}else{Node* up = parent->_parent;if (up->_kv.first > sub->_kv.first)up->_left = sub;elseup->_right = sub;sub->_parent = up;}parent->_right = subL;if (subL != nullptr)subL->_parent = parent;sub->_left = parent;parent->_parent = sub;parent->_bf = 0;sub->_bf = 0;return sub;
}

c. 新节点插入较高左子树的右侧---左右:先左单旋再右单旋 

Node* RotateLR(Node* parent)
{Node* child = parent->_left;Node* sub = child->_right;Node* subL = sub->_left;Node* subR = sub->_right;if (parent == _root){_root = sub;sub->_parent = nullptr;}else{Node* up = parent->_parent;if (up->_kv.first > sub->_kv.first)up->_left = sub;elseup->_right = sub;sub->_parent = up;}sub->_right = parent;parent->_parent = sub;sub->_left = child;child->_parent = sub;parent->_left = subR;if (subR != nullptr)subR->_parent = parent;child->_right = subL;if (subL != nullptr)subL->_parent = child;if (sub->_bf == 1){parent->_bf = 0;child->_bf = -1;}else if (sub->_bf == -1){parent->_bf = 1;child->_bf = 0;}else{parent->_bf = 0;child->_bf = 0;}sub->_bf = 0;return sub;
}

d. 新节点插入较高右子树的左侧---右左:先右单旋再左单旋

Node* RotateRL(Node* parent)
{Node* child = parent->_right;Node* sub = child->_left;Node* subL = sub->_left;Node* subR = sub->_right;if (parent == _root){_root = sub;sub->_parent = nullptr;}else{Node* up = parent->_parent;if (up->_kv.first > sub->_kv.first)up->_left = sub;elseup->_right = sub;sub->_parent = up;}sub->_left = parent;parent->_parent = sub;sub->_right = child;child->_parent = sub;parent->_right = subL;if (subL != nullptr)subL->_parent = parent;child->_left = subR;if (subR != nullptr)subR->_parent = child;if (sub->_bf == 1){parent->_bf = -1;child->_bf = 0;}else if (sub->_bf == -1){parent->_bf = 0;child->_bf = 1;}else{parent->_bf = 0;child->_bf = 0;}sub->_bf = 0;return sub;
}

四、AVL树删除

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

具体实现可参考《算法导论》或《数据结构-用面向对象方法与C++描述》殷人昆版。

附录:AVL树实现参考代码 

AVLTree · 梁羽赫/cpp_advanced - 码云 - 开源中国 (gitee.com)icon-default.png?t=N7T8https://gitee.com/yuhe-liang/cpp_advanced/tree/master/AVLTree

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

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

相关文章

IIS 服务器,下载APK 文件,用于发布更新最新的APK包

IIS 默认情况下无法下载 .apk 文件&#xff0c;需要对 IIS 服务进行设置 1、打开 IIS 对应的应用 选中MIME 类型 右键 打开功能 2、右键添加 文件扩展名&#xff1a;.apk MIME 类型输入&#xff1a;application/vnd.android.package-archive 3、重启应用 4、浏览器访问 服务地…

OpenMV学习笔记4——二维码识别

一、示例程序 按照下图顺序点击&#xff0c;即可打开官方在IDE中准备好的二维码实例程序&#xff1a; # QRCode Example # # This example shows the power of the OpenMV Cam to detect QR Codes # using lens correction (see the qrcodes_with_lens_corr.py script for hig…

SRE养成计划(持续更新)

SRE&#xff08;Site Reliability Engineering&#xff0c;网站可靠性工程师&#xff09;是一种实践&#xff0c;旨在将软件工程的方法和工具应用于运维领域&#xff0c;以提高大型复杂系统的可靠性、可扩展性和性能。SRE工程师不仅负责保持系统的高可用性和性能&#xff0c;还…

局域网电脑监控软件是如何监控到内网电脑的?

在信息化快速发展的今天&#xff0c;局域网电脑监控软件成为许多企业、学校和机构重要的实用工具。这些软件的主要功能在于对局域网内的电脑进行实时监控&#xff0c;以确保网络的安全、员工的工作效率以及合规性。那么&#xff0c;局域网电脑监控软件是如何做到对内网电脑进行…

后端开发面经系列 -- 华为C++一面面经

HUAWEI – C一面面经 公众号&#xff1a;阿Q技术站 来源&#xff1a;https://www.nowcoder.com/feed/main/detail/b8113ff340d7444985b32a73c207c826 1、计网的协议分几层&#xff1f;分别叫什么&#xff1f; OSI七层模型 物理层 (Physical Layer): 负责物理设备之间的原始比…

苹果手机数据不见了怎么恢复?3个方法,搞定苹果手机数据恢复!

在许多错误的情况下&#xff0c;当你更新到最新的 iOS 版本或使用越狱来获得更多功能和权限、误删重要的手机文件时&#xff0c;苹果手机中的数据可能会丢失或被意外删除。一旦发现数据丢失&#xff0c;你就会查看 iTunes 备份或 iCloud 备份&#xff0c;并希望在其中恢复丢失的…

纷享销客安全体系: 组织及人员安全

组织及人员安全是纷享销客安全战略中的重要组成部分。 我们致力于确保组织内部和员工的安全&#xff0c;并采取一系列措施来预防和应对安全威胁。我们将持续改进和更新安全措施&#xff0c;以适应不断变化的威胁环境&#xff0c;并确保组织和员工的安全意识和培训得到充分关注…

【LeetCode 35】搜索插入位置

1. 题目 2. 分析 经典的二分搜索题&#xff0c;没有特别之处。可以用于考察同学对二分法的理解是否到位。这道题的关键点在于&#xff1a;找到满足第一个大于等于target条件的数的下标即可&#xff0c;而这个下标就是left。 所以最后返回值就是left。 3. 代码 class Soluti…

inBuilder 低代码平台新特性推荐 - 第二十期

今天来给大家带来的是 inBuilder 低代码平台特性推荐系列第二十期——菜单导航模式个性化示例。 场景介绍 目前平台提供了四种菜单导航模式&#xff0c;包括分组视图、列表视图、横向视图、平铺视图&#xff0c;均为横向导航&#xff0c;这些也是主流的菜单导航模式。 在某些…

05-控制流(分支结构)

05-控制流(分支结构) 一、二路分支 程序中某一段代码需要满足一定的条件才会被执行。 if 语句&#xff1a;用于表达一种条件&#xff0c;如果条件满足则执行某个代码块。if-else 语句&#xff1a;用于表达一种条件&#xff0c;如果条件满足则执行某个代码块&#xff0c;否则…

css文本超长溢出显示省略号 ...

1、单行文本溢出省略 对于单行文本&#xff0c;可以使用以下CSS样式来实现溢出省略号显示&#xff1a; .single-line-ellipsis { width: 100px; /* 设置容器宽度 */ overflow: hidden; /* 隐藏超出容器的内容 */ text-overflow: ellipsis; /* 用省略号表示溢出的文本 */ whit…

车载以太网测试要测些什么呢?

车载以太网测试大致可以分成两块&#xff1a;TC8测试和以太网通信测试。 TC8测试全称TC8一致性测试&#xff0c;其规范由OPEN联盟制定&#xff0c;包括车载以太网ECU从物理层到应用层的各层互操作性以及常规基础功能服务。目的在于提高不同ECU之间的兼容性。 TC8测试规范可以…

基于Golang的AOI算法实现与优化

本文深入探讨了AOI&#xff08;Area of Interest&#xff09;算法在游戏开发中的重要性&#xff0c;详细介绍了如何使用Golang语言实现AOI算法&#xff0c;包括网格法、四叉树、十字链表等实现方式&#xff0c;并讨论了性能优化策略及实际应用场景。 文章目录 AOI算法概述AOI算…

Springboot项目:使用MockMvc测试get和post接口(含单个和多个请求参数场景)

一、引入MockMvc依赖 使用MockMvc&#xff0c;必须要引入依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-test</artifactId><scope>test</scope></dependency>二、具体演示…

差分原理+练习

差分的原理和前缀和相似&#xff0c;我们先联想一下前缀和。 前缀和计算下标从0到n的和&#xff0c;记为sum[n1];如果想要求出[l,r]区间的和&#xff0c;可以快速的通过sum[r1]-sum[l]来得到 。 前缀和适用于需要多次获取指定连续区间和的情景 而差分即计算相邻两个元素的差…

搜索与图论:树的重心

搜索与图论&#xff1a;树的重心 题目描述参考代码 题目描述 输入样例 9 1 2 1 7 1 4 2 8 2 5 4 3 3 9 4 6输出样例 4参考代码 #include <cstring> #include <iostream> #include <algorithm>using namespace std;const int N 100010, M N * 2;int n, m…

在Vue中使用websocket的流程

在Vue中使用WebSocket的完整流程可以总结如下&#xff1a; 安装WebSocket库&#xff08;如果需要&#xff09; 如果你选择使用特定的Vue WebSocket库&#xff08;如vue-websocket&#xff09;&#xff0c;你可以通过npm进行安装。 npm install vue-websocket --save或者&#…

【VUE 具名插槽的应用】

具名插槽类似于提前将布局安排好&#xff0c;但内容为空&#xff0c;一旦有具体内容填充进来&#xff0c;可以很和谐的展示&#xff0c;不影响整体效果。&#x1f347; “举个&#x1f330;&#xff1a;系统里大部分页面的查询条件是相同的&#xff0c;所以需要封装一个公用的查…

【2024最新华为OD-C/D卷试题汇总】目录管理器 (200分) - 支持在线评测+三语言AC题解(Python/Java/Cpp)

🍭 大家好这里是清隆学长 ,一枚热爱算法的程序员 ✨ 本系列打算持续跟新华为OD-C/D卷的三语言AC题解 💻 ACM银牌🥈| 多次AK大厂笔试 | 编程一对一辅导 👏 感谢大家的订阅➕ 和 喜欢💗 在线评测链接 目录管理器 (200分) 🌍 评测功能需要订阅专栏后私信联系清隆解…

【Python Cookbook】S01E23 以不区分大小写的方式对文本做查找和替换 re.IGNORECASE

目录 问题解决方案讨论 问题 如果我们需要从一段字符串中以不区分大小写的方式做查找和替换&#xff0c;该怎么做&#xff1f; 解决方案 正则化模块 re 提供 re.IGNORECASE 标识从而实现以不区分大小写的方式做查找和替换。 查找 re.findall() import retext "UPPER…