AVL树(Insert)

news2024/11/24 22:44:19

文章目录

  • AVL树
    • 代码
    • 模拟实现
      • Insert重点
          • 控制平衡:
        • 旋转->平衡
          • 右单旋
          • 左单旋
          • 左右双旋
          • 双旋平衡因子更新错误
            • 右左双旋
            • 左右双旋
      • erase了解
    • AVL树的性能

AVL树

代码

高度平衡二叉搜索树

  • 引入

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

于在顺序表中搜索元素,效率低下。因此,两位俄罗斯的数学家G.M.Adelson-Velskii和E.M.Landis在1962年

发明了一种解决上述问题的方法:当向二叉搜索树中插入新结点后,如果能保证每个结点的左右子树高度之

差的绝对值不超过1(需要对树中的结点进行调整),即可降低树的高度,从而减少平均搜索长度。AVL树用三个人名首字母命名的。

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

    • 它的左右子树都是AVL树

    • 左右子树高度之差(简称平衡因子)的绝对值不超过1(-1/0/1)

      • 不一定需要平衡因子,使用平衡因子是一种控制实现方式

非常接近满二叉树,插入10亿个节点将近30层左右.

模拟实现

Insert重点

在SB树的基础之上如何保证接近平衡呢>-平衡因子.一个节点的平衡因子受到他的左右子树的影响。

  • 插入到节点的右边,该节点的平衡因子需要++,左边就–.
  • 新增节点只会影响他的祖先节点的平衡因子.
控制平衡:
  1. 更新平衡因子,新增节点到根节点的祖先路径

  2. 出现异常平衡因子就需要旋转平衡树

  3. 出现一下五种情况:

  • 更新以后,parent->bf == 0,更新结束。说明更新前parent->bf是1或者-1,现在变成0,说明填上了矮的那边,parent所处的子树高度不变也就不会对上一层进行影响。
  • 更新以后,parent->bf==1/-1,继续向上更新,更新前parent->bf=0,现在变成1、-1,我有一遍子树变高了
    parent所在的子树高度变了,需要继续向上更新.
  • 更新以后,parent->bf=2或者-2,parent已经不平衡了,需要旋转处理。根据平衡因子的正负来区别类型。

简单情况推演:

image-20230301201910640

旋转->平衡

右单旋
  • 右单旋引入

image-20230301211033321

右单旋(左腿长)

  1. 保持搜索树的规则

  2. 控制平衡

  • 右单旋抽象图

image-20230301210938276

  • 右单旋具象图

image-20230301211333066

  • 右单旋代码理解

image-20230301213111066

	void RotateR(Node* parent)
	{
		Node* subl = parent->_left;
		Node* sublr = subl->_right;
		//进行旋转维护三叉连
		parent->_left = sublr;
		if (sublr)
			sublr->_parent = parent;
		Node* pparent = parent->_parent;

		subl->_right = parent;
		parent->_parent = subl;

		if (parent == _root)
		{
			_root = subl;
			_root->_parent = nullptr;
		}
		else
		{
			if (pparent->_left == parent)
				pparent->_left = subl;
			else
				pparent->_right = subl;
			subl->_parent = pparent;
		}
		subl->_parent = parent->_bf = 0;
	}
  • 连接时要注意是否是空节点,以及一定要保证三叉连。
左单旋

右边腿长

右单旋

	void RotateL(Node* parent)
	{
		Node* subr = parent->_right;
		Node* subrl = subr->_left;

		Node* pparent = parent->_parent;
		parent->_parent = subr;
		subr->_left = parent;
		parent->_right = subrl;
		if (subrl)
			subrl->_parent = parent;
		if (parent == _root)
		{
			_root = subr;
			subr->_parent == nullptr;
		}

		else
		{
			if (pparent->_left == parent)
			{
				pparent->_left = subr;
			}
			else
			{
				pparent->_right = subr;
			}
			subr->_parent = pparent;
		}
		subr->_bf = parent->_bf = 0;
	}
左右双旋

image-20230302092416929

  • 如何验证AVL树是没有问题的?高度,每一颗子树都得检查一遍是不是AVL树的形状。IsBalance()

  • 平衡因子是否都更新正确?树构建完成之后再进行插入节点。

双旋平衡因子更新错误
右左双旋

双旋时平衡因子的更新是由两次单旋完成的,在这里出现的问题,在未是平衡树的时候就将平衡因子更新为0了。

image-20230302100708934

  • 如何识别是那种情况?

    可以通过60 的平衡因子识别三种情况,所以在两次旋转之前要将三个特殊节点保存

	void RotateRL(Node* parent)
	{
		Node* subr = parent->_right;
		Node* subrl = subr->_left;
		int bf = subrl->_bf;

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

		if (bf == 1)
		{
			parent->_bf = -1;
		}
		else if (bf == -1)
		{
			subr->_bf = 1;
		}
		else if (bf == 0)
		{
			//
		}
		else
		{
			assert(false);
		}
	}
左右双旋

对三种情况特判来更新平衡因子即可.

image-20230302110113195

void RotateLR(Node* parent)
	{
		Node* subl = parent->_left;
		Node* sublr = subl->_right;
		int bf = sublr->_bf;

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

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

所以要在双旋的函数中,在单旋操作对于平衡因子更新之后,再进行对于平衡因子特殊情况的自我更新。对于值的更新不明白的自己画个图就明白了,-1 1
0都是确定的,因为只有出发这些情况才会满足。性能接近logN.

erase了解

首先按照搜索树的规则删除,然后更新平衡因子,如果有需要再进行旋转.

image-20230302111036957

AVL树的性能

AVL树是一棵绝对平衡的二叉搜索树,其要求每个节点的左右子树高度差的绝对值都不超过1,这样可以保证

查询时高效的时间复杂度,即 logN.但是如果要对AVL树做一些结构修改的操作,性能非常低下,比如:

插入时要维护其绝对平衡,旋转的次数比较多,更差的是在删除时,有可能一直要让旋转持续到根的位置。

因此:如果需要一种查询高效且有序的数据结构,而且数据的个数为静态的(即不会改变),可以考虑AVL树,

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

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

相关文章

C/C++网络编程笔记Socket

https://www.bilibili.com/video/BV11Z4y157RY/?vd_sourced0030c72c95e04a14c5614c1c0e6159b上面链接是B站的博主教程,源代码来自上面视频,侵删,这里只是做笔记,以供复习和分享。上一篇博客我记录了配置环境并且跑通了&#xff0…

从头开始搭建一个SpringBoot项目--SpringBoot文件的上传与下载

从头开始搭建一个SpringBoot项目--SpringBoot文件的上传前言流程分析代码结构代码详情UploadFileInfo.classUploadController.classUploadDao.classUploadDao.xmlUploadServices.classUploadServicesImpl.class测试下载示例前言 文件的上传和下载是很多系统必备的功能&#xf…

pytorch-复现经典深度学习模型-LeNet5

Neural Networks 使用torch.nn包来构建神经网络。nn包依赖autograd包来定义模型并求导。 一个nn.Module包含各个层和一个forward(input)方法,该方法返回output。 一个简单的前馈神经网络,它接受一个输入,然后一层接着一层地传递,…

代码随想录算法训练营day47 |动态规划 198打家劫舍 213打家劫舍II 337打家劫舍III

day47198.打家劫舍1.确定dp数组(dp table)以及下标的含义2.确定递推公式3.dp数组如何初始化4.确定遍历顺序5.举例推导dp数组213.打家劫舍II情况一:考虑不包含首尾元素情况二:考虑包含首元素,不包含尾元素情况三&#x…

网络技术|网络地址转换与IPv6|路由设计基础|4

对应讲义——p6 p7NAT例题例1解1例2解2例3解3例4解4一、IPv6地址用二进制格式表示128位的一个IPv6地址,按每16位为一个位段,划分为8个位段。若某个IPv6地址中出现多个连续的二进制0,可以通过压缩某个位段中的前导0来简化IPv6地址的表示。例如…

1月奶粉电商销售数据榜单:销售额约20亿,高端化趋势明显

鲸参谋电商数据监测的2023年1月份京东平台“奶粉”品类销售数据榜单出炉! 根据鲸参谋数据显示,1月份京东平台上奶粉的销量约675万件,销售额约20亿元,环比均下降19%左右。与去年相比,整体也下滑了近34%。可以看出&#…

真无线耳机哪个牌子好用?2023便宜好用的无线耳机推荐

蓝牙耳机经过近几年的快速发展,变得越来越普及,并且在一些性能上也做得越来越好。那么,真无线耳机哪个牌子好用?下面,我来给大家推荐几款便宜好用的无线耳机,可以参考一下。 一、南卡小音舱蓝牙耳机 参考…

Nuxt 3.0 全栈开发:五种渲染模式的差异和使用场景全解析

Nuxt 3.0 全栈开发 - 杨村长 - 掘金小册核心知识 工程架构 全栈进阶 项目实战,快速精通 Nuxt3 开发!。「Nuxt 3.0 全栈开发」由杨村长撰写,299人购买https://s.juejin.cn/ds/S6p7MVo/ 前面我们提到过 Nuxt 能够满足我们更多开发场景的需求…

IGKBoard(imx6ull)-I2C接口编程之SHT20温湿度采样

文章目录1- 使能开发板I2C通信接口2- SHT20硬件连接3- 编码实现SHT20温湿度采样思路(1)查看sht20从设备地址(i2cdetect)(2)获取数据大体流程【1】软复位【2】触发测量与通讯时序(3)返…

日志收集笔记(Kibana,Watcher)

1 Kibana Kibana 是一个开源的分析与可视化平台,可以用 Kibana 搜索、查看存放在 Elasticsearch 中的数据,就跟谷歌的 elasticsearch head 插件类似,但 Kibana 与 Elasticsearch 的交互方式是各种不同的图表、表格、地图等,直观的…

【python】控制台中文输出乱码解决方案

注:最后有面试挑战,看看自己掌握了吗 文章目录控制台原因解决方法方法一方法二方法三如果是os.system函数乱码控制台原因 一般的情况下,还是我们的源码文件的编码格式问题。我们一般是要把源码文件的编码格式改成utf-8就好了,但是…

zeppelin安装及hive配置

一、zeppelin安装包 链接:https://pan.baidu.com/s/1DVmvY2TM7WmCskejTn8dzA 提取码:fl7r 二、安装zeppelin 将安装包传入Centos的/opt/install目录下 # 解压 tar -zxf /opt/install/zeppelin-0.10.0-bin-all.tgz -C /opt/soft/ # 重命名 mv /opt/sof…

Nodejs环境配置 | Linux安装nvm | windows安装nvm

文章目录一. 前言二. Linux Nodejs环境配置1. 安装nvm2. 配置npm三. Windows Nodejs环境配置1. 安装nvm2. 配置npm四. nvm基本使用一. 前言 由于在实际开发中一些不同的项目需要不同的npm版本来启动,所以本篇文章会基于nvm这个node版本管理工具来进行Linux和Winodw…

[AI助力] 2022.3.2 考研英语学习 2011 英语二翻译

[AI助力] 2022.3.2 考研英语学习 2011 英语二翻译 文章目录[AI助力] 2022.3.2 考研英语学习 2011 英语二翻译2011年英语二翻译真题自己写的看看AI的翻译谷歌翻译New Bing😂让AI自我评价chatgpt🤣让AI自我评价DeepL有道腾讯翻译百度翻译IDEA翻译积累&…

智能家居项目(八)之树莓派+摄像头进行人脸识别

目录 1、编辑Camera.c 2、编辑contrlDevices.h 3、编辑mainPro.c 4、进行编译: 5、运行结果: ./test1 6、项目图片演示 智能家居项目(七)之Libcurl库与HTTPS协议实现人脸识别_Love小羽的博客-CSDN博客 经过上一篇文章&…

redhawk:Low Power Analysis

1.rush current与switch cell 在standby状态下为了控制leakage power我们选择power gating的设计方式,使用power switch cell关闭block/power domain的电源。 power switch的基本介绍可见: 低功耗设计-Power Switch power switch的table中有四种状态,…

Simulink 自动代码生成电机控制:优化Simulink生成的代码提升代码运行效率

目录 优化需求 优化方法 从模型配置优化 优化结果对比 从算法层优化 优化结果对比 总结 优化需求 本次优化的目的是提升FOC代码执行速度,以普通滑模观测器为例,优化前把速度环控制放到2ms的周期单独运行,把VOFA上位机通信代码放到主循…

mongodb入门到使用(上)

mongodb的安装与使用前言一、linux下载二、mongodb配置三、 mongodb服务管理启动服务查看停止四、远程连接五、SpringBoot整合总结前言 本文主要针对一些项目的部署服务器在使用方面用到了mongodb,参考解决一些部署方面遇到的问题。 一、linux下载 使用wget下载 w…

代数小课堂:向量代数(通过向量夹角理解不同的维度)

文章目录 引言I 计算向量的夹角1.1 毕达哥拉斯定理1.2 余弦定理1.3 计算向量的夹角II 向量夹角的应用2.1 用计算机自动筛选简历(对人进行分类)2.2 计算机进行文本自动分类的原理引言 根据余弦定理计算两个向量的夹角向量夹角的应用: 对文本进行自动分类、自动筛选简历。如果…

【上位机入门常见问题】Visual Studio 2022安装指导

Visual Studio 2022安装指导 这里给大家指导安装Visual Studio 2022 Community版本,也就是我们常说的社区版,这个版本是微软给开发者学习技术专门定制的免费版本,其他的专业版和企业版都是商业收费版本。对于我们学习,大家使用社…