AVL 树

news2025/1/16 8:04:05

目录

  • AVL树的概念
  • AVL树节点的定义
  • AVL树的插入
  • AVL树的旋转
    • 左单旋(parent->_bf == 2 && cur->_bf == 1)
      • a,b,c当高度为0
      • a,b,c当高度为1
      • a,b,c当高度为2
      • a,b,c当高度为......
    • 右单旋(parent->_bf == -2 && cur->_bf == -1)
      • a,b,c当高度为0
      • a,b,c当高度为1
      • a,b,c当高度为2
      • a,b,c当高度为........
  • 左右双旋(parent->_bf == -2 && cur->_bf == 1)
      • a,b,c当高度为0
      • a,b,c当高度为1
      • a,b,c当高度为2
      • a,b,c当高度为.......
  • 右左双旋(parent->_bf == 2 && cur->_bf == -1)
      • a,b,c当高度为0
      • a,b,c当高度为1
      • a,b,c当高度为2
      • a,b,c当高度为.......
  • AVL树的性能

AVL树的概念

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

为什么左右高度差不是相等呢?

因为有些结点数量是永远不能保证高度差相等的比如2个,4个… 所以只能退而求其次,左右高度差不超过1.。

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

  • 它的左右子树都是AVL树
  • 左右子树高度之差(简称平衡因子)的绝对值不超过1(-1/0/1)
  • 每个子树也都要遵守左右子树高度之差的绝对值不超过1
  • AVL树的实现不一定必须要用平衡因子,平衡因子只是它的一种实现方式,个人感觉平衡因子比较容易理解AVL树

在这里插入图片描述
如果一棵二叉搜索树是高度平衡的,它就是AVL树。如果它有n个结点,其高度可保持在 O ( l o g 2 n ) O(log_2 n) O(log2n),搜索时间复杂度O( l o g 2 n log_2 n log2n)。
一个正常的平衡二叉搜索树的的平衡因子只能是-1 0 1

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)
		:_left(nullptr)
		,_right(nullptr)
		,_parent(nullptr)
		,_kv(kv)
		,_bf(0)//新插入的结点,没有左右子树,所以平衡因子为0
	{}
};

AVL树的插入

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

  1. 按照二叉搜索树的方式插入新节点
  2. 更新平衡因子
  3. 如果更新完以后,平衡因为没有出现问题 ∣ b f ∣ < = 1 |bf| <= 1 bf<=1,平衡结构没有受到影响,不需要处理
  4. 如果更新完以后,平衡出现问题 ∣ b f ∣ > 1 |bf|> 1 bf>1,平衡结构受到影响, 需要处理(旋转)

在这里插入图片描述
🔍 插入新增节点,会影响祖先的平衡因子(全部或者部分)

  1. 新增结点在parent左边 cur == parent->_left) parent->_bf--
  2. 新增结点在parent右边cur == parent->_right parent->_bf++

🔍 每增加一个结点,就要检查parent所在的子树高度是否变化? 变了继续更新,不变则不再更新。

  1. 新增结点后,如果parent的平衡因子等于1或者-1,parent所在的子树高度变了,需要继续向上更新平衡因子。为什么呢?因为在插入结点之前parent的平衡因子等于0,说明插入之前左右两个子树高度相等,现在parent一边高一边低,高度发生变化需要继续向上更新平衡因子。
  2. 新增结点后,如果parent的平衡因子等于2或者-2,parent所在的子树高度不平衡了,需要处理这个子树(进行旋转)。
  3. 新增结点后,如果parent的平衡因子等于0,parent所在的子树高度不变,不需要处理继续向上更新,这一次插入结束。为什么呢?说明插入之前parent的平衡因子等于1或者-1,插入之前parent一边高一边低,插入的结点填在了矮的那一边,它的高度不变。
bool Insert(const pair<K, V>& kv)
{
	if (nullptr == _root)//第一次插入
	{
		_root = new Node(kv);
		return true;
	}
	//插入的树不是空树
	Node* parent = nullptr;
	Node* cur = _root;

	while (cur)
	{
		if (cur->_kv.first < kv.first)
		{
			parent = cur;
			cur = cur->_right;
		}
		else if (cur->_kv.first > kv.first)
		{
			parent = cur;
			cur = cur->_left;
		}
		else
		{
			//不支持重复数据
			return false;
		}
	}
	//插入数据
	cur = new Node(kv);
	if (parent->_kv.first < cur->_kv.first)
	{
		parent->_right = cur;
	}
	else
	{
		parent->_left = cur;
	}
	//链接父结点
	cur->_parent = parent;

	//更新平衡因子
	//父结点的平衡因子 = 左子树高度 - 右子树高度

	while (parent)//终止条件,根结点的父结点为空
	{
		if (cur == parent->_left)
		{
			parent->_bf--;
		}
		else
		{
			parent->_bf++;
		}

		if (parent->_bf == 1 || parent->_bf == -1)
		{
			parent = parent->_parent;
			cur = cur->_parent;
		}
		else if (parent->_bf == 0)
		{
			break;
		}
		else if (parent->_bf == 2 || parent->_bf == -2)
		{
			// 需要旋转处理 -- 1、让这颗子树平衡 2、降低这颗子树的高度
			if (parent->_bf == 2 && cur->_bf == 1)
			{
				RotateL(parent);
			}
			else if (parent->_bf == -2 && cur->_bf == -1)
			{
				RotateR(parent);
			}
			else if (parent->_bf == -2 && cur->_bf == 1)
			{
				RotateLR(parent);
			}
			else if (parent->_bf == 2 && cur->_bf == -1)
			{
				RotateRL(parent);
			}
			else
			{
				//说明这个AVL树存在问题
				assert(false);
			}

			break;
		}
		else
		{
			//说明这个AVL树存在问题
			assert(false);
		}
	}

	return true;
}

AVL树的旋转

💡旋转的原则:保持他继续是搜索树。
💡旋转的目的:左右均衡,降低整棵树的高度。
🔍如果在一棵原本是平衡的AVL树中插入一个新节点,可能造成不平衡,此时必须调整树的结构,使之平衡化。根据节点插入位置的不同,AVL树的旋转分为四种:

左单旋(parent->_bf == 2 && cur->_bf == 1)

在这里插入图片描述

a,b,c当高度为0

在这里插入图片描述

a,b,c当高度为1

在这里插入图片描述

a,b,c当高度为2

在这里插入图片描述

在这里插入图片描述

a,b,c当高度为…

后面的情况就不画了情况太多了。只要符合parent->_bf == 2 && cur->_bf == 1的子树和根都可以进行左单旋。
💻左单旋代码

//右子树高,进行左单旋
void RotateL(Node* parent)
{
	Node* childR = parent->_right;
	Node* childRL = childR->_left;

	parent->_right = childRL;
	if (childRL)//如果childRL不为空,要链接父节点
		childRL->_parent = parent;

	Node* pparent = parent->_parent;
	childR->_left = parent;
	parent->_parent = childR;

	if (nullptr == pparent)//说明是根结点
	{
		_root = childR;
		//_root->_parent = nullptr;
		childR->_parent = nullptr;
	}
	else
	{
		//判断是pparent的左子树还是右子树
		if (pparent->_left == parent)
		{
			pparent->_left = childR;
		}
		else
		{
			pparent->_right = childR;
		}

		childR->_parent = pparent;//链接父结点
	}

	parent->_bf = childR->_bf = 0;//更新平衡因子
}

右单旋(parent->_bf == -2 && cur->_bf == -1)

在这里插入图片描述

a,b,c当高度为0

在这里插入图片描述

a,b,c当高度为1

在这里插入图片描述

a,b,c当高度为2

在这里插入图片描述
在这里插入图片描述

a,b,c当高度为…

后面的情况就不画了情况太多了。只要符合parent->_bf == -2 && cur->_bf == -1的子树和根都可以进行右单旋。
💻右单旋代码

//左子树高,进行右单旋
void RotateR(Node* parent)
{
	Node* childL = parent->_left;
	Node* childLR = childL->_right;

	parent->_left = childLR;
	if (childLR)
		childLR->_parent = parent;

	Node* pparent = parent->_parent;
	childL->_right = parent;
	parent->_parent = childL;

	if (_root == parent)//说明是根结点
	{
		_root = childL;
		_root->_parent = nullptr;
	}
	else
	{
		//判断是pparent的左子树还是右子树
		if (pparent->_left == parent)
		{
			pparent->_left = childL;
		}
		else
		{
			pparent->_right = childL;
		}

		childL->_parent = pparent;//链接父结点
	}

	parent->_bf = childL->_bf = 0;//更新平衡因子
}

左右双旋(parent->_bf == -2 && cur->_bf == 1)

在这里插入图片描述

a,b,c当高度为0

在这里插入图片描述

a,b,c当高度为1

在这里插入图片描述

a,b,c当高度为2

在这里插入图片描述

a,b,c当高度为…

后面的情况就不画了情况太多了。只要符合parent->_bf == -2 && cur->_bf == 1`的子树和根都可以进行左右双旋。
💻左右双旋代码

//左子树的右子树高,进行左右双旋
void RotateLR(Node* parent)
{
	Node* childL = parent->_left;
	Node* childLR = childL->_right;
	int bf = childLR->_bf;//bf有三种情况

	RotateL(parent->_left);
	RotateR(parent);

	if (bf == 1)//bf == 1代表新增加结点是childLR的左子树
	{
		parent->_bf = 0;
		childLR->_bf = 0;
		childL->_bf = -1;
	}
	else if (bf == -1)//bf == -1代表新增加结点是childLR的右子树
	{
		parent->_bf = 1;
		childLR->_bf = 0;
		childL->_bf = 0;
	}
	else if (bf == 0)//bf == 0代表新增加结点是childLR,使平衡因子发生错误
	{
		parent->_bf = 0;
		childLR->_bf = 0;
		childL->_bf = 0;
	}
	else
	{
		assert(false);
	}
	
}

右左双旋(parent->_bf == 2 && cur->_bf == -1)

在这里插入图片描述

a,b,c当高度为0

在这里插入图片描述

a,b,c当高度为1

在这里插入图片描述

a,b,c当高度为2

在这里插入图片描述

a,b,c当高度为…

后面的情况就不画了情况太多了。只要符合parent->_bf == 2 && cur->_bf == -1的子树和根都可以进行右左双旋。

💻右左双旋代码

//右子树的左子树高,进行右左双旋
void RotateRL(Node* parent)
{
	Node* childR = parent->_right;
	Node* childRL = childR->_left;
	int bf = childRL->_bf;//bf有三种情况

	RotateR(parent->_right);
	RotateL(parent);

	if (bf == 1)
	{
		parent->_bf = -1;
		childR->_bf = 0;
		childRL->_bf = 0;
	}
	else if (bf == -1)
	{
		parent->_bf = 0;
		childR->_bf = 1;
		childRL->_bf = 0;
	}
	else if (bf == 0)
	{
		parent->_bf = 0;
		childR->_bf = 0;
		childRL->_bf = 0;
	}
	else
	{
		assert(false);
	}
}

AVL树的性能

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

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

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

相关文章

强化学习从基础到进阶-案例与实践[4]:深度Q网络-DQN、double DQN、经验回放、rainbow、分布式DQN

【强化学习原理项目专栏】必看系列&#xff1a;单智能体、多智能体算法原理项目实战、相关技巧&#xff08;调参、画图等、趣味项目实现、学术应用项目实现 专栏详细介绍&#xff1a;【强化学习原理项目专栏】必看系列&#xff1a;单智能体、多智能体算法原理项目实战、相关技巧…

Azkaban初认识

Azkaban初认识 文章目录 Azkaban初认识Azkaban是什么&#xff1f;为什么需要工作流调度系统&#xff1f;常见的工作流调度系统Azkaban 与 Oozie的对比 Azkaban是什么&#xff1f; Azkaban是一个开源的分布式工作流管理器&#xff0c;在LinkedIn实施&#xff0c;以解决Hadoop作业…

RT-Thread-03-栈空间分配

栈空间分配 线程状态转换图&#xff1a; 系统滴答时钟 每个操作系统都存在一个系统时钟&#xff0c;是操作系统中最小的时钟单位。这个时钟负责系统和时间相关的一些操作。这个时钟由硬件定时器的定时中断产生。 系统时钟的频率需要根据芯片的处理能力来决定&#xff0c; 频…

【MySQL基础 | 第一篇】数据处理之基本查询

前言 查询语句属于DML&#xff08;Data Manipulation Language&#xff09;数据操作语言的其中一种&#xff0c;用于从数据库中提取所需的数据。通过灵活的条件和组合&#xff0c;查询语句帮助用户有效地获取、过滤和排序数据&#xff0c;满足各种信息需求。 文章目录 前言1️⃣…

团体程序设计天梯赛-练习集L1篇⑨

&#x1f680;欢迎来到本文&#x1f680; &#x1f349;个人简介&#xff1a;Hello大家好呀&#xff0c;我是陈童学&#xff0c;一个与你一样正在慢慢前行的普通人。 &#x1f3c0;个人主页&#xff1a;陈童学哦CSDN &#x1f4a1;所属专栏&#xff1a;PTA &#x1f381;希望各…

编译原理笔记17:自下而上语法分析(4)LR(0)、SLR(1) 分析表的构造

目录 LR(0) 文法LR(0) 分析表的构造例 SLR(1) 文法SLR 分析表构造 非 SLR(1) 文法举例二义文法都不是 SLR(1) 文法不是二义文法的非 SLR(1) 文法 LR(0) 文法 若一个文法 G 的拓广文法 G’ 的识别活前缀的自动机中的每个状态&#xff08;项目集&#xff09;均不存在下述情况&…

【一文通】C/C++与Go语言混合编程入门级教程(Windows平台完成)

一、概述 Go语言可以通过自带的 cgo 工具进行 CGO 混合编程&#xff0c;这个工具放在go安装目录的 pkg\tool 下&#xff0c;其源代码则在 src\runtime\cgo 里面&#xff0c;当然作为入门教程本文不打算对cgo的实现原理进行深入研究&#xff0c;仅从 Hello World 的角度来实际体…

快速查询银行卡发卡省市和归属银行,了解自己的财务状况!

API接口是现代软件开发的基本组成部分。它们允许应用程序通过互联网连接到其他软件系统&#xff0c;并从这些系统中获取或传输数据。银行卡归属地查询API接口是为开发人员提供的一种工具&#xff0c;可以帮助他们轻松地查询银行卡的归属地信息。在本文中&#xff0c;我们将介绍…

内网横向移动—IPCATschtasks

内网横向移动—IPC&AT&schtasks 1. IPC介绍1.1. IPC利用条件关系1.1.1. IPC$空连接介绍1.1.2. 139与445端口介绍1.1.3. 默认共享 1.2. IPC连接问题1.2.1. IPC连接失败原因1.2.2. IPC连接常见错误 2. 横向移动常用命令2.1. IPC命令介绍2.1.1. IPC常用命令演示2.1.1.1. 建…

docker非root用户下取消sudo前缀

解决非root用户下执行docker命令提示权限不足&#xff0c;必须添加sudo的问题。 第一步&#xff1a;执行 sudo gpasswd aby docker 命令&#xff0c;将当前用户aby加入docker组中。 第二步&#xff1a;执行 sudo chmod arw /var/run/docker.sock 命令修改sock权限

数据结构之堆——算法与数据结构入门笔记(六)

本文是算法与数据结构的学习笔记第六篇&#xff0c;将持续更新&#xff0c;欢迎小伙伴们阅读学习。有不懂的或错误的地方&#xff0c;欢迎交流 引言 当涉及到高效的数据存储和检索时&#xff0c;堆&#xff08;Heap&#xff09;是一种常用的数据结构。上一篇文章中介绍了树和完…

chatgpt赋能python:Python列表转字符串——从新手到大师

Python列表转字符串——从新手到大师 在Python编程中&#xff0c;列表和字符串是非常常用的数据类型。有时候&#xff0c;我们需要将一个列表转换为一个字符串&#xff0c;以方便进行各种操作。幸运的是&#xff0c;Python内置了一些函数和方法&#xff0c;可以轻松地将列表转…

7Z045 引脚功能详解

本文针对7Z045芯片&#xff0c;详细讲解硬件设计需要注意的技术点&#xff0c;可以作为设计和检查时候的参考文件。问了方便实用&#xff0c;按照Bank顺序排列&#xff0c;包含配置Bank、HR Bank、HP Bank、GTX Bank、供电引脚等。 参考文档包括&#xff1a; ds191-XC7Z030-X…

ruoyi-cloud版本(一)项目的下载与本地运行(亲测有效)

目录 1 架构2 架构图3 源码下载4 创建数据库5 下载nacos与运行6 打开运行基础模块&#xff08;启动没有先后顺序&#xff09;7 启动前端 1 架构 com.ruoyi ├── ruoyi-ui // 前端框架 [80] ├── ruoyi-gateway // 网关模块 [8080] ├── ruoyi…

canvas详解00-认识canvas

身为一个WEB开发人员&#xff0c;肯定都是想着能够开发出酷炫和激动人心的应用程序来。可以很多动画特效&#xff0c;例如黑客帝国的数字&#xff0c;彩色炫酷的例子动效。也可以实现各种图画面板&#xff0c;如实现类似于photoshop的web在线图像编辑。各种酷炫的表单等等。 #…

专项练习10

目录 一、选择题 1、执行以下程序&#xff0c;下列说法中&#xff0c;正确的是&#xff08;&#xff09; 2、下面有关JavaScript中系统方法的描述&#xff0c;错误的是&#xff1f; 3、以下 JavaScript 代码&#xff0c;在浏览器中运行的结果是 4、假设DOM结构为 二、编程题 …

[ruby on rails] rails中使用graphQL

1. 添加gem gem graphql’是主要提供server的, gem graphiql-rails’是用来生成一个graphiql查询页面IDE,自己用来测试的group :development dogem graphiql-rails endgem graphql2.使用命令生成模板文件 rails g graphql:install在API only中,routes不会自动填充graphiql路…

chatgpt赋能python:Python的下载方法——从官网到第三方渠道

Python的下载方法——从官网到第三方渠道 Python 是一种翻译式、面向对象的、动态数据类型的高级程序设计语言&#xff0c;被广泛应用于数据分析、人工智能、物联网等领域。相信大多数程序员都知道 Python&#xff0c;并且使用它编写程序。那么&#xff0c;如何下载 Python&am…

人工智能(1):机器学习工作流程

1 什么是机器学习 机器学习是从数据中自动分析获得模型&#xff0c;并利用模型对未知数据进行预测。 2 机器学习工作流程 机器学习工作流程总结 1 获取数据 2 数据基本处理 3 特征工程 4 机器学习(模型训练) 5 模型评估 结果达到要求&#xff0c;上线服务没有达到要求&a…

程序编译连接加载过程详解

程序加载过程详解 可重定位的elf文件格式简介 首先我们打开目标文件看一下 上面的图就是目标文件的格式了&#xff0c;这里使用的是010editer&#xff0c;这个二进制编辑器很好用 可以看到大致分为三部分&#xff0c;首先是header&#xff0c;然后是sectionheader&#xff0…