二叉平衡搜索树-AVL树

news2025/1/15 17:58:20

目录

  • 1. avl树的概念
  • 2. 树结点的定义
  • 3. 结点的插入
    • 3.1 左单旋
    • 3.2 右单旋
    • 3.3 右左双旋
    • 3.4 左右双旋
  • 4. 结点的删除(了解)
  • 5. 整体代码

1. avl树的概念

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

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

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

平衡因子的计算一般是右子树的高度-左子树的高度

因此AVL树也叫做二叉平衡树,若树上有n个结点,树的高度可保持在log2N,搜索的次数也就是它的高度次。

2. 树结点的定义

template<class K, class V>
struct AVLTreeNode {
	AVLTreeNode(pair<const K, V>& kv)
		:_kv(kv), _bf(0), _left(nullptr), _right(nullptr), _parent(nullptr)
	{

	}
	pair<const K, V> _kv;	    //key/value模型
	int _bf;					//平衡因子
	AVLTreeNode<K, V>* _left;
	AVLTreeNode<K, V>* _right;
	//除了左右指针外还要增加一个指向父节点的指针
	//因为后续插入结点后需要向上调整祖先们的平衡因子
	AVLTreeNode<K, V>* _parent;	
};

3. 结点的插入

avl树插入的第一步和二叉搜索树是一致的,根据搜索树的性质选择一个合适的位置进行插入,不同的是插入完成后,树的平衡性可能会受到破坏,因此需要更新祖先们的平衡因子,若插入在左边parent的平衡因子-1,反之+1,更新后会有三种情况:

  1. 平衡因子==0
    这种情况说明parent这颗(子)树的整体高度不变,因此不需要再沿着祖先们的路径往上进行更新。
  2. 平衡因子==1 || -1
    这种情况说明parent这颗(子)树的高度发生了变化,因此需要继续沿着祖先们的路径往上进行更新。
  3. 平衡因子==2 || -2
    这种情况说明parent这颗(子)树的高度发生变化的同时且不再满足平衡,需要对其进行旋转让其变得平衡。

对于情况三,该如何进行旋转?这里需要分类讨论:

在这里插入图片描述
在旋转的时候需要注意的是:

  1. 旋转后依然保持它是搜索树
  2. 变成平衡树的同时降低子树的高度

3.1 左单旋

新节点插入较高右子树的右侧—右右:左单旋:
在这里插入图片描述
左旋的核心在于要把cur的左子树交给parent的右子树,然后parent作为cur的左子树,旋转完成后还需要修改对应结点父指针的指向和平衡因子。

代码实现:

void rotateLeft(Node* parent) {
	//对于左单旋只与三个结点有关
	//parent、cur(parent->_right)和cur->left
	Node* cur = parent->_right;
	Node* curleft = cur->_left;
	//cur一定不为空而cur->left是有可能为空的
	//因此需要特殊判断
	if (curleft) {
		//指向新parent
		curleft->_parent = parent;
	}
	//核心操作
	parent->_right = curleft;
	cur->_left = parent;	

	//先保存parent的父节点后续会用到
	//然后修改对应父节点的指向
	Node* oldparent = parent->_parent; 
	parent->_parent = cur;
	cur->_parent = oldparent;

	//parent可能是根节点也可能是一颗子树
	//若是根节点那么oldparent则为空
	//cur为新的根
	if (!oldparent) {
		_root = cur;
	}
	//否则判断parent结点是oldparent的哪颗子树
	//让其指向新的孩子cur
	else {
		if (oldparent->_left == parent) {
			oldparent->_left = cur;
		}
		else {
			oldparent->_right = cur;
		}
	}
	//旋转完毕后与插入之前(子)树的高度是不变的,同时还填上了矮的那颗子树
	//因此要修改cur和parent的平衡因子为0
	cur->_bf = parent->_bf = 0;
}

3.2 右单旋

新节点插入较高左子树的左侧—左左:右单旋:
在这里插入图片描述
右单旋的逻辑与左单旋非常相似,该逻辑的核心操作在于要把cur的右子树交给parent的左子树,让parent作为cur的右子树,同时旋转完成后 修改对应结点父指针的指向和平衡因子。

代码实现:

void ratateRight(Node* parent) {
	//同样只与三个结点有关
	//parent、cur(parent->_left)和cur->_right
	Node* cur = parent->_left;
	Node* curright = cur->_right;
	if (curright) {
		curright->_parent = parent;
	}
	parent->_left = curright;
	cur->_right = parent;

	Node* oldparent = parent->_parent;
	parent->_parent = cur;
	cur->_parent = oldparent;

	if (!oldparent) {
		_root = cur;
	}
	else {
		if (oldparent->_left == parent) {
			oldparent->_left = cur;
		}
		else {
			oldparent->_right = cur;
		}
	}

	cur->_bf = parent->_bf = 0;
}

3.3 右左双旋

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

单旋只能解决纯粹是一边高的场景(左左高或者右右高),对于下面这种场景单旋无法解决:
在这里插入图片描述
类似上述这种更为复杂的树结构也是同样的道理,因此要用到双旋解决。

抽象点说若需要旋转的那颗(子)树的结构呈一条直线,那么单旋就可以解决,而若为折线的情况,则要用双旋

对于折线情况,能否先让其"变"成一条直线再进行单旋呢?这里的变就是第一次单旋,旋转为直线后再进行第二次单旋。

第一次旋转如何操作?对于上图的例子,其实就是先以cur为旋转点对cur和新插入的结点进行右单旋(对于这两个结点而言是左边高),捋直了后就变成了纯粹的一边高,此时再以parent为旋转点对parent和cur进行左单旋:
在这里插入图片描述
这是最简单的一种情况,稍微复杂点的树结构比如下图,也是需要通过双旋来解决:
在这里插入图片描述
若插入再40结点的右边时旋转后的结构会有什么不同呢?
在这里插入图片描述
可以发现的是若插入在40的左边时,旋转完成后新节点成了parent的右子树,而插入在右边时最终变成了cur的左子树,parent和cur分别为40这个结点的左右子树,40成了这棵树的根。

对于其它更为复杂的树结构,如下抽象图,解决方法也是同样的道理:
在这里插入图片描述

双旋结束后,还需要更新对应结点的平衡因子,根据上面的三个例子,parent和cur平衡因子会有三种情况:

  1. parent和cur都为0。
  2. parent为0,cur为1。
  3. parent为-1,cur为0。

如何区分的关键在于40这个结点是否是新增;在40结点的左边插入;在40结点的右插入。

若平衡因子是第一种情况,那么40是新增结点;若是第二种情况,那么是在40的左边插入新节点;最后一种情况是在40的右边插入新结点。

代码实现:

void rotatRightLeft(Node* parent) {
	//先保存关键结点的平衡因子
	Node* cur = parent->_right;
	Node* curleft = cur->_left;
	int bf = curleft->_bf;

	//复用单旋
	rotateRight(cur);
	rotateLeft(parent);

	//旋转后根据情况更新对应的平衡因子
	//自己就是新增
	if (bf == 0) {
		parent->_bf = cur->_bf = curleft->_bf = 0;
	}
	//在左边插入
	else if (bf == -1) {
		parent->_bf = curleft->_bf = 0;
		cur->_bf = 1;
	}
	//右边插入
	else if (bf == 1) {
		cur->_bf = curleft->_bf = 0;
		parent->_bf = -1;
	}
	else {
		assert(false);
	}
}

3.4 左右双旋

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

左右双旋的情况简单概括为如下三种:
在这里插入图片描述
对于左右双旋情况的整体分析是基本与右左双旋一致,直接上代码:

void rotatLeftRight(Node* parent) {
	Node* cur = parent->_left;
	Node* curright = cur->_right;
	int bf = curright->_bf;

	rotateLeft(cur);
	rotateRight(parent);
	
	if (bf == 0) {
		parent->_bf = cur->_bf = curright->_bf = 0;
	}
	//在左边插入
	else if (bf == -1) {
		cur->_bf = curright->_bf = 0;	 
		parent->_bf = 1;
	}
	//右边插入
	else if (bf == 1) {
		parent->_bf = curright->_bf = 0;
		cur->_bf = -1;
	}
	else {
		assert(false);
	}
}

4. 结点的删除(了解)

因为AVL树也是二叉搜索树,可按照二叉搜索树的方式将节点删除,然后再更新平衡因子,只不错与删除不同的时,删除节点后的平衡因子更新,最差情况下一直要调整到根节点的位置,并且调整的过程相比于插入会更加复杂,所以稍微了解就好。

5. 整体代码

#pragma once
#include <iostream>
#include <assert.h>
#include <windows.h>


using namespace std;

template<class K, class V>
struct AVLTreeNode {
	AVLTreeNode(const pair<const K, V>& kv)
		:_kv(kv), _bf(0), _left(nullptr), _right(nullptr), _parent(nullptr)
	{

	}
	pair<const K, V> _kv;	    //key/value模型
	int _bf;					//平衡因子
	AVLTreeNode<const K, V>* _left;
	AVLTreeNode<const K, V>* _right;
	//除了左右指针外还要增加一个指向父节点的指针
	//因为后续插入结点后需要向上调整祖先们的平衡因子
	AVLTreeNode<const K, V>* _parent;
};

template<class K, class V>
class AVLTree {
	typedef AVLTreeNode<const K, V> Node;
public:
	bool insert(const pair<const K, V>& kv) {
		//首先根据搜索树的性质找到一个合适的位置进行插入
		if (!_root) {
			_root = new Node(kv);
			return true;
		}
		Node* cur = _root, *parent = _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 < kv.first) {
			parent->_right = cur;
		}
		else {
			parent->_left = cur;
		}
		cur->_parent = parent;
		//插入完毕后需要沿着祖先路径调整平衡因子

		//最多调整到根
		while (parent) {
			if (parent->_left == cur) {
				--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) {
				//分别对几种旋转的情况进行判断

				//右边高需要左单旋
				if (parent->_bf == 2 && cur->_bf == 1) {
					rotateLeft(parent);
				}
				//左边高需要右单旋
				else if (parent->_bf == -2 && cur->_bf == -1) {
					rotateRight(parent);
				}
				//右左双旋
				else if (parent->_bf == 2 && cur->_bf == -1) {
					rotatRightLeft(parent);
				}
				//左右双旋
				else if (parent->_bf == -2 && cur->_bf == 1) {
					rotatLeftRight(parent);
				}
				else {
					assert(false);
				}
				break;
			}
			else {
				assert(false);
			}
		}
		return true;
	}
	void getSingleTreeHeight() {
		_getSingleTreeHeight(_root);
	}
	
private:
	void _getSingleTreeHeight(Node* root) {
		if (root) {
			printf("树%d的高度为:%d, 左子树为%d,右子树为%d\n\n",  
				root->_kv.first, getHeight(root), getHeight(root->_left), getHeight(root->_right));
			_getSingleTreeHeight(root->_left);
			_getSingleTreeHeight(root->_right);
		}
	}
	int getHeight(Node* root) {
		if (!root) {
			return 0;
		}
		int leftH = getHeight(root->_left);
		int rightH = getHeight(root->_right);
		return max(leftH, rightH) + 1;
	}
	void rotateLeft(Node* parent) {
		//对于左单旋只与三个结点有关
		//parent、cur(parent->_right)和cur->left
		Node* cur = parent->_right;
		Node* curleft = cur->_left;
		//cur一定不为空而cur->left是有可能为空的
		//因此需要特殊判断
		if (curleft) {
			//指向新parent
			curleft->_parent = parent;
		}
		//核心操作
		parent->_right = curleft;
		cur->_left = parent;	

		//先保存parent的父节点后续会用到
		//然后修改对应父节点的指向
		Node* oldparent = parent->_parent; 
		parent->_parent = cur;
		cur->_parent = oldparent;

		//parent可能是根节点也可能是一颗子树
		//若是根节点那么oldparent则为空
		//cur为新的根
		if (!oldparent) {
			_root = cur;
		}
		//否则判断parent结点是oldparent的哪颗子树
		//让其指向新的孩子cur
		else {
			if (oldparent->_left == parent) {
				oldparent->_left = cur;
			}
			else {
				oldparent->_right = cur;
			}
		}
		//旋转完毕后与插入之前(子)树的高度是不变的,同时还填上了矮的那颗子树
		//因此要修改cur和parent的平衡因子为0
		cur->_bf = parent->_bf = 0;
	}

	void rotateRight(Node* parent) {
		//右单旋的逻辑与左单旋非常相似
		// 
		//同样只与三个结点有关
		//parent、cur(parent->_left)和cur->_right
		Node* cur = parent->_left;
		Node* curright = cur->_right;
		if (curright) {
			curright->_parent = parent;
		}
		parent->_left = curright;
		cur->_right = parent;

		Node* oldparent = parent->_parent;
		parent->_parent = cur;
		cur->_parent = oldparent;

		if (!oldparent) {
			_root = cur;
		}
		else {
			if (oldparent->_left == parent) {
				oldparent->_left = cur;
			}
			else {
				oldparent->_right = cur;
			}
		}

		cur->_bf = parent->_bf = 0;
	}

	//右左双旋
	void rotatRightLeft(Node* parent) {
		//先保存关键结点的平衡因子
		Node* cur = parent->_right;
		Node* curleft = cur->_left;
		int bf = curleft->_bf;

		//复用单旋
		rotateRight(cur);
		rotateLeft(parent);

		//旋转后根据情况更新对应的平衡因子
		//自己就是新增
		if (bf == 0) {
			parent->_bf = cur->_bf = curleft->_bf = 0;
		}
		//在左边插入
		else if (bf == -1) {
			parent->_bf = curleft->_bf = 0;
			cur->_bf = 1;
		}
		//右边插入
		else if (bf == 1) {
			cur->_bf = curleft->_bf = 0;
			parent->_bf = -1;
		}
		else {
			assert(false);
		}
	}

	//左右双旋
	void rotatLeftRight(Node* parent) {

		Node* cur = parent->_left;
		Node* curright = cur->_right;
		int bf = curright->_bf;

		rotateLeft(cur);
		rotateRight(parent);

		if (bf == 0) {
			parent->_bf = cur->_bf = curright->_bf = 0;
		}
		//在左边插入
		else if (bf == -1) {
			cur->_bf = curright->_bf = 0;	 
			parent->_bf = 1;
		}
		//右边插入
		else if (bf == 1) {
			parent->_bf = curright->_bf = 0;
			cur->_bf = -1;
		}
		else {
			assert(false);
		}
	}
	Node* _root = nullptr;
};

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

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

相关文章

C++基础——类与对象

1 概述 C是面向对象的语言&#xff0c;面向对象语言三大特性&#xff1a;封装、继承、多态。 C将万事万物抽象为对象&#xff0c;对象上有其属性和行为。 2 封装 2.1 封装的意义 封装是面向对象的三大特性之一&#xff0c;封装将属性和行为作为一个整体&#xff0c;对属性和…

顺丰函证通API集成,无代码开发连接CRM和电商平台

1. 顺丰&#xff1a;全球第四大快递公司的无代码开发连接 顺丰是全球第四大快递公司&#xff0c;秉承 “以用户为中心&#xff0c;以需求为导向&#xff0c;以体验为根本” 的产品设计思维。顺丰不仅在国内市场深耕&#xff0c;而且横向拓展多元业务领域&#xff0c;纵深完善产…

Node Sass version 9.0.0 is incompatible with ^4.0.0.

1.错误产生原因&#xff1a; node、 node-sass 和sass-loader的版本对应问题 2.解决方案&#xff1a; 删除之前的 npm uninstall node-sass sass-loader 安装指定的 npm i node-sass4.14.1 sass-loader7.3.1 --save -dev

业绩持续增长,“创新与变革”是云南白药发展的不二法门?

提及云南白药&#xff0c;大多数消费者的第一反应便是云南白药气雾剂、云南白药牙膏等产品。事实上&#xff0c;随着消费需求驱动、行业升级走向愈发明确&#xff0c;云南白药早已启动从传统中药制造企业到现代化大健康企业的转型&#xff0c;并持续产出成果。 近日&#xff0…

Kubernetes技术与架构-存储 4

如上所示&#xff0c;Kubernetes集群支持动态申请存储资源&#xff0c;即集群管理员可以按照实际的需求动态地申请存储资源&#xff0c;集群管理员需要事先定义一个或者多个StorageClass存储类型的资源&#xff0c;Pod中的容器实例直接引用事先定义的StorageClass存储类型的资源…

开关电源泄漏电流测试方法| 万用表测量开关电源漏电流的方法及接线方式分享

漏电流测试是开关电源安规测试项目之一&#xff0c;目的是为了检测漏电流是否超过了额定标准&#xff0c;防止漏电流过大造成设备损毁&#xff0c;甚至引发电击安全事故。漏电流测试方法多样&#xff0c;纳米软件将带你了解如何用万用表测量开关电源的漏电流。 开关电源漏电流测…

Squid

一、Squid 代理服务器 Squid 主要提供缓存加速、应用层过滤控制的功能。 二、代理的工作机制 1&#xff0e;代替客户机向网站请求数据&#xff0c;从而可以隐藏用户的真实IP地址。 2&#xff0e;将获得的网页数据&#xff08;静态 Web 元素&#xff09;保存到缓存中并发送给…

关于 国产系统UOS系统Qt开发Tcp服务器外部连接无法连接上USO系统 的解决方法

若该文为原创文章&#xff0c;转载请注明原文出处 本文章博客地址&#xff1a;https://hpzwl.blog.csdn.net/article/details/134254817 红胖子(红模仿)的博文大全&#xff1a;开发技术集合&#xff08;包含Qt实用技术、树莓派、三维、OpenCV、OpenGL、ffmpeg、OSG、单片机、软…

接口自动化测试分层设计与实践总结01

本文以笔者当前使用的自动化测试项目为例&#xff0c;浅谈分层设计的思路&#xff0c;不涉及到具体的代码细节和某个框架的实现原理&#xff0c;重点关注在分层前后的使用对比&#xff0c;可能会以一些伪代码为例来说明举例。 接口测试三要素&#xff1a; 参数构造 发起请求&…

【Redis】SSM整合Redis注解式缓存的使用

&#x1f389;&#x1f389;欢迎来到我的CSDN主页&#xff01;&#x1f389;&#x1f389; &#x1f3c5;我是Java方文山&#xff0c;一个在CSDN分享笔记的博主。&#x1f4da;&#x1f4da; &#x1f31f;推荐给大家我的专栏《Redis》。&#x1f3af;&#x1f3af; &#x1f4…

2000-2022年上市公司供应链数字化示范名单匹配数据

2000-2022年上市公司供应链数字化示范名单匹配数据 1、时间&#xff1a;2000-2022年 2、来源&#xff1a;商务部 3、指标&#xff1a; 上市公司供应链数字化&#xff08;根据城市名单匹配&#xff09;&#xff1a;股票代码、年份、股票简称、中文全称、省份、城市、区县、上…

接口请求断言

接口请求断言是指在发起请求之后&#xff0c;对返回的响应内容去做判断&#xff0c;用来查看是否响应内容是否与规定的返回值相符。 在发起请求后&#xff0c;我们使用一个变量 r 存储响应的内容&#xff0c;也就是 Response 对象。 Response 对象有很多功能强大的方法可以调…

[动态规划] (九) 路径问题:LeetCode 64.最小路径和

[动态规划] (九) 路径问题&#xff1a;LeetCode 64.最小路径和 文章目录 [动态规划] (九) 路径问题&#xff1a;LeetCode 64.最小路径和题目解析解题思路状态表示状态转移方程初始化和填表顺序返回值 代码实现总结 64. 最小路径和 题目解析 (1) 从左上角到右下角 (2) 只能向右…

Poetry:Python开发者的依赖管理新时代

更多资料获取 &#x1f4da; 个人网站&#xff1a;涛哥聊Python 在Python开发中&#xff0c;管理项目的依赖关系是一个至关重要的任务。传统上&#xff0c;开发者使用requirements.txt文件和pip工具来管理依赖&#xff0c;但这种方式在复杂项目中存在一些问题。Poetry是一个现…

Docker 学习路线 5:在 Docker 中实现数据持久化

Docker 可以运行隔离的容器&#xff0c;包括应用程序和其依赖项&#xff0c;与主机操作系统分离。默认情况下&#xff0c;容器是临时的&#xff0c;这意味着容器中存储的任何数据在终止后都将丢失。为了解决这个问题并在容器生命周期内保留数据&#xff0c;Docker 提供了各种数…

kafka问题汇总

报错1&#xff1a; 解决方式 1、停止docker服务   输入如下命令停止docker服务 systemctl stop docker 或者service docker stop1   停止成功的话&#xff0c;再输入docker ps 就会提示出下边的话&#xff1a; Cannot connect to the Docker daemon. Is the docker daem…

通过全流量查看部门或客户端网络使用情况

近年来&#xff0c;随着数字化转型和云计算服务的广泛应用&#xff0c;组织和企业对于网络带宽和性能的需求也在不断增长。 网络的稳定性、性能和安全性对于业务流程的顺畅运行至关重要。因此&#xff0c;了解部门或客户端网络的使用情况是网络管理和优化的关键。本文将通过Ne…

【C++】STL容器适配器——stack类的使用指南(含代码使用)(17)

前言 大家好吖&#xff0c;欢迎来到 YY 滴C系列 &#xff0c;热烈欢迎&#xff01; 本章主要内容面向接触过C的老铁 主要内容含&#xff1a; 欢迎订阅 YY滴C专栏&#xff01;更多干货持续更新&#xff01;以下是传送门&#xff01; 目录 一、stack 类——基本介绍二、stack 类…

if语句中的按位取反问题

&#x1f380; 文章作者&#xff1a;二土电子 &#x1f338; 关注公众号获取更多资源&#xff01; &#x1f438; 期待大家一起学习交流&#xff01; 文章目录 一、现象描述1.1 在C语言中&#xff08;非STM32&#xff09;1.2 STM32中运行 二、基础知识复习2.1 原码、反码和补…

dell r720部署chatglm3,使用nvidia tesla P40+M40

dell r720的idrac的地址默认是192.168.1.110&#xff0c;root 默认密码calvin fatal Error! All channnels have been disabled due to all DIMMs failed the Memoey 是什么意思 Dell PowerEdge T320服务器 开机显示 Fatal Errort!all channells have been disabled due to …