【C++笔记】AVL树

前言

各位读者朋友们大家好,上期我们讲解了map和set这两大容器的使用,这一期我们讲解最早的平衡二叉搜索树------AVL树。

目录

  • 前言
  • [一. AVL树的概念](#一. AVL树的概念)
  • [二. AVL树的实现](#二. AVL树的实现)
    • [2.1 AVL树的结构](#2.1 AVL树的结构)
    • [2.2 AVL树的插入](#2.2 AVL树的插入)
      • [2.2.1 AVL树插入一个值的大致过程](#2.2.1 AVL树插入一个值的大致过程)
      • [2.2.2 平衡因子的更新](#2.2.2 平衡因子的更新)
      • [2.2.3 插入代码实现](#2.2.3 插入代码实现)
    • [2.3 旋转](#2.3 旋转)
      • [2.3.1 旋转的原则](#2.3.1 旋转的原则)
      • [2.3.2 右单旋(单纯左子树高)](#2.3.2 右单旋(单纯左子树高))
      • [2.3.3 右单旋的实现](#2.3.3 右单旋的实现)
      • [2.3.4 左单旋(单纯右子树高)](#2.3.4 左单旋(单纯右子树高))
      • [2.3.5 左单旋的实现](#2.3.5 左单旋的实现)
      • [2.3.6 左右双旋(左子树的右子树高)](#2.3.6 左右双旋(左子树的右子树高))
      • [2.3.7 左右双旋的实现](#2.3.7 左右双旋的实现)
      • [2.3.8 右左双旋(右子树的左子树高)](#2.3.8 右左双旋(右子树的左子树高))
      • [2.3.9 右左双旋的实现](#2.3.9 右左双旋的实现)
    • [2.4 AVL树的查找](#2.4 AVL树的查找)
  • [三. AVL树的全体实现](#三. AVL树的全体实现)
    • [3.1 AVL树的平衡检测](#3.1 AVL树的平衡检测)
  • 结语

一. AVL树的概念

  • AVL树是最先发明的自平衡二叉查找树,AVL是一棵空树,或者具备下列性质的⼆叉搜索树:它的左右子树都是AVL树,且左右子树的高度差的绝对值不超过1。AVL树是一颗高度平衡搜索二叉树,通过控制高度差去控制平衡。
  • AVL树得名于它的发明者G.M.Adelson-Velsky和E.M.Landis是两个前苏联的科学家,他们在1962年的论文中发表了它。
  • AVL树实现这里我们引入一个平衡因子(balance factor)的概念,每个节点都有一个平衡因子,任何节点的平衡因子等于右子树的高度减去左子树的高度 ,也就是说任何节点的平衡因子等于0/1/-1,AVL树并不是必须要平衡因子,但是有了平衡因子可以更方便我们去进行观察和控制树是否平衡,就像一个风向标一样。
  • AVL树整体节点数量和分布和完全二叉树类似,高度可控制在logN,增删查改的效率也可以控制在O (logN),相比二叉搜索树有了本质的提升。

二. AVL树的实现

2.1 AVL树的结构

cpp 复制代码
template <class K, class V>
struct AVLTreeNode
{
	pair<K, T> _kv;
	AVLTreeNode* _left;
	AVLTreeNode* _right;
	AVLTreeNode* _parent;
	int _bf;// 平衡因子

	AVLTreeNode(const pair<K,V>& kv)
		:_kv(kv)
		,_left(nullptr)
		,_right(nullptr)
		,_parent(nullptr)
		,_bf(0)
	{}
};
template <class K,class V>
class AVLTree
{
	typedef AVLTreeNode<K, V> Node;
public:

private:
	Node* _root = nullptr;
};

2.2 AVL树的插入

2.2.1 AVL树插入一个值的大致过程

  1. 按照二叉搜索树的规则插入一个值
  2. 新增节点以后,只会影响祖先节点的高度,也就是可能会影响部分祖先节点的平衡因子,所以更新插入节点到根路径上的平衡因子,实际中最坏的情况下要更新到根,有些情况更新到中间就可以停止了,具体情况下面再详细分析。
  3. 更新平衡因子过程中没有出现问题,插入结束。
  4. 更新平衡因子过程中出现不平衡,对不平衡子树旋转,旋转后调平衡的同时,本质降低了子树的高度,不会再影响上一层,所以插入结束。

2.2.2 平衡因子的更新

更新原则:

  • 平衡因子 = 右子树高度 - 左子树高度

  • 只有子树高度变化才会影响当前节点的平衡因子

  • 插入节点,会增加高度,所以新增加节点在parent的右子树,parent的平衡因子++,在左子树,平衡因子--

  • parent所在子树的高度是否变化决定了是否向上继续更新
    更新停止条件:

  • 更新后parent的平衡因子等于0,更新中parent的平衡因子的变化为1到0或者-1到0,说明更新前parent的子树一边高一边低,新插入的节点在低的一侧,插入后parent所在子树的高度不变,不会影响parent父亲节点的平衡因子,更新结束。

    这种情况下就相当于将parent子树补齐

  • 更新后parent的平衡因子等于1或者-1,更新前,更新中parent的平衡因子的变化为0到1或者0到-1,说明更新前parent的左右子树同高,新增的插入节点后,parent所在子树一边高一边低,parent所在子树的符合平衡要求,但是高度增加了1,会影响parent的平衡因子,所以要继续向上调整。当更新到根节点的时候结束更新

  • 更新后parent的平衡因子等于2或者-2,parent的平衡因子的变化为1到2或者-1到-2,说明更新前子树一边高一边低,然后新插入的节点在高的那边,parent所在子树高的那边更高了,破坏了平衡,parent所在的子树不符合平衡要求,需要旋转处理。旋转的目标有两个:1、把parent的子树旋转平衡。 2、降低parent子树的高度,恢复到插入节点之前的高度 。所以旋转之后也不需要向上更新,插入结束。

2.2.3 插入代码实现

cpp 复制代码
bool Insert(const pair<K, V>& kv)
{
	if (_root == nullptr)
	{
		_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 < kv.first)
	{
		parent->_right = cur;
	}
	else
	{
		parent->_left = cur;
	}
	// 链接父亲
	cur->_parent = parent;
	// 更新平衡因子
	while (parent)
	{
		if (cur == parent->_right)// 插入右边平衡因子++
			++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)
		{
			// 旋转
		}
		else
			assert(false);
	}
	return true;
}

插入前面的逻辑根二叉搜索树插入的逻辑一样,只是插入完节点之后,更新了平衡因子,并调整树的平衡。

2.3 旋转

2.3.1 旋转的原则

  • 保持搜索树的规则
  • 让旋转的树从不平衡变得平衡,其次降低树的高度
    旋转一共有四种:左单旋/右单旋/左右双旋/右左双旋

2.3.2 右单旋(单纯左子树高)

旋转核心步骤,因为5 < b子树的值 < 10,将b变成10的左子树,10变成5的右子树,5变成这棵树新的根,符合搜索树的规则,控制了平衡,同时这棵树的高度恢复到了插入之前的h+2,符合旋转原则。如果插入之前10整棵树的⼀个局部子树,旋转后不会再影响上一层,插入结束了。

上图中是抽象图,有a/b/c抽象为三棵高度为h的子树(h>=0),a/b/c均符合AVL树的要

求。10可能是整棵树的根,也可能是⼀个整棵树中局部的子树的根。这里a/b/c是高度为h的子树,上图适用于所有的右单旋情况,下面我们展开几种情况来看一下:

  • 插入前a/b/c/的高度为0
  • 插入前a/b/c/的高度为1
  • 插入前a/b/c/的高度为2


    a必须是x,否则就不能在10节点旋转了(可能在5的左孩子节点旋转,也可能不会旋转)
  • 插入前a/b/c/的高度为3


    只有a是4个节点或者三个节点插入到两个节点上才会旋转。

2.3.3 右单旋的实现

subL当根节点,parent当subL的右子树,subLR当parent的右子树,也要注意所有的parent的处理,每个节点要将对应的父亲节点调整好

cpp 复制代码
void RotateR(Node* parent)
{
	// subL当根节点,parent当subL的右子树,subLR当parent的右子树,注意所有的parent的处理
	Node* subL = parent->_left;
	Node* subLR = subL->_right;
	subL->_right = parent;
	parent->_left = subLR;
	// 从下往上链接父节点
	if (subLR)// subL可能为空,为空就不需要处理父亲节点
		subLR->_parent = parent;
	Node* pParent = parent->_parent;// 父节点的父亲,便于后续链接
	parent->_parent = subL;
	if (parent == _root)// 如果调整的是整棵树的根
	{
		_root = subL;
		subL->_parent = nullptr;
	}
	else// 局部树,链接祖先
	{
		if (parent == pParent->_left)
			pParent->_left = subL;
		else
			pParent->_right = subL;
		subL->_parent = pParent;
	}
	// 更新平衡因子
	subL->_bf = parent->_bf = 0;
}

2.3.4 左单旋(单纯右子树高)

  • a/b/c抽象为三棵高度为h的子树(h>=0),a/b/c均符合AVL树的要求。10可能是整棵树的根,也可能是一整棵树中局部的子树的根。这里a/b/c是高度为h的子树,是一种概括抽象表示,他代表了所有左单旋的场景,实际左单旋形态有很多种,和右单旋很类似。
  • 在a子树中插入一个新节点,导致a子树的高度从h变成h+1,不断向上更新平衡因子,导致10的平衡因子从1变成2,10为根的树左右高度差超过1,违反平衡规则。10为根的树右边太高了,需要往左边旋转,控制两棵树的平衡。
  • 旋转核心步骤,因为10 < b子树的值 < 15,将b变成10的右子树,10变成15的左子树,15变成这棵树新的根,符合搜索树的规则,控制了平衡,同时这棵的高度恢复到了插入之前的h+2,符合旋转原则。如果插入之前10整棵树的一个局部子树,旋转后不会再影响上一层,插入结束了。

2.3.5 左单旋的实现

cpp 复制代码
void RotateL(Node* parent)
{
	Node* subR = parent->_right;
	Node* subRL = subR->_left;

	parent->_right = subRL;
	subR->_left = parnet;
	// 从下往上链接父节点
	if (subRL)
		subRL->_parent = parent;
	Node* pParent = parent->_parent;
	parent->_parent = subR;
	if (parent == _root)// 调整的是整棵树的根
	{
		_root = subR;
		subR->_parent = nullptr;
	}
	else
	{
		if (parent == pParent->_left)
			pParent->_left = subR;
		else
			pParent->_right = subR;
		subR->_parent = pParent;
	}
	subR->_bf = parent->_bf = 0;
}

2.3.6 左右双旋(左子树的右子树高)

我们先通过具体图来分析:

  • 情况1、a/b/c子树的高度为0
  • 情况2、a/b/c子树的高度h==1,在b树的右边插入节点
  • 情况3、a/b/c子树的高度h==1,在b树的左边插入节点

2.3.7 左右双旋的实现

cpp 复制代码
void RotateLR(Node* parent)
{
	Node* subL = parent->_left;
	Node* subLR = subL->_right;
	int bf = subLR->_bf;// 记录其平衡因子,看新节点是插入在左还是在右
	RotateL(subL);// 先左旋
	RotateR(parent);// 再右旋
	// 更新平衡因子
	if (bf == 1)// 在右侧插入的节点
	{
		parent->_bf = 0;
		subL->_bf = -1;
		subLR->_bf = 0;
	}
	else if (bf == -1)// 左侧
	{
		parent->_bf = 1;
		subL->_bf = 0;
		subLR->_bf = 0;
	}
	else if (bf == 0)// 树为空直接插入
	{
		parent->_bf = 0;
		subL->_bf = 0;
		subLR->_bf = 0;
	}
	else
		assert(false);
}

2.3.8 右左双旋(右子树的左子树高)

这里和左右双旋一样,也是有三种情况

2.3.9 右左双旋的实现

cpp 复制代码
void RotateRL(Node * parent)
{
	Node* subR = parent->_right;
	Node* subRL = subR->_left;
	int bf = subRL->_bf; // 记录其平衡因子,看新节点是插入在左还是在右
	// 先右旋
	RotateR(subR);
	// 左旋
	RotateL(parent);
	if (bf == -1)// 左边插入 
	{
		parent->_bf = 0;
		subR->_bf = 1;
		subRL->_bf = 0;
	}
	else if (bf == 1)
	{
		parent->_bf = -1;
		subR->_bf = 0;
		subRL->_bf = 0;
	}
	else if (bf == 0)
	{
		parent->_bf = 0;
		subR->_bf = 0;
		subRL->_bf = 0;
	}
	else
		assert(false);
}

2.4 AVL树的查找

和二叉树搜索树的逻辑一样,只是效率提升到O(logN)

cpp 复制代码
Node* Find(const K& key)
{
	Node* cur = _root;
	while (cur)
	{
		if (key > cur->_kv.first)// 大的往右走
			cur = cur->_right;
		else if (key < cur->_kv.first)// 小的往左走
			cur = cur->_left;
		else
			return cur;
	}
	return nullptr;
}

三. AVL树的全体实现

AVL树的实现代码

3.1 AVL树的平衡检测

我们可以检查AVL树的高度差,如果高度差>=2,就说明不是AVL树

cpp 复制代码
bool _IsBalanceTree(Node * root)
	{
		if (root == nullptr)
			return true;
		int leftHeight = _Height(root->_left);
		int rightHeight = _Height(root->_right);
		int diff = rightHeight - leftHeight;
		if (abs(diff) >= 2)
		{
			cout << root->_kv.first << "高度差异常,不是AVL树!!!" << endl;
			return false;
		}
		if (diff != root->_bf)
		{
			cout << root->_kv.first << "平衡因子异常" << endl;
			return false;
		}
		return _IsBalanceTree(root->_left) && _IsBalanceTree(root->_right);
	}

测试代码

cpp 复制代码
void TestAVLTree2()
{
	const int N = 100000;
	vector<int> v;
	v.reserve(N);
	srand(time(0));
	for (size_t i = 0; i < N; i++)
	{
		v.push_back(rand() + i);
	}
	size_t begin2 = clock();
	AVLTree<int, int> t;
	for (auto e : v)
	{
		t.Insert(make_pair(e, e));
	}
	size_t end2 = clock();
	cout << "Insert:" << end2 - begin2 << endl;
	if (t.IsBalanceTree())
		cout << "AVL" << endl;
	cout << "Height:" << t.Height() << endl;
	cout << "Size:" << t.Size() << endl;
	size_t begin1 = clock();
	for (size_t i = 0; i < N; i++)
	{
		t.Find((rand() + i));
	}
	size_t end1 = clock();
	cout << "Find:" << end1 - begin1 << endl;
}


int main()
{
	TestAVLTree2();
	return 0;
}

结语

以上我们就讲完了AVL树的实现,这部分需要好好消化,感谢大家的阅读!欢迎大家批评指正!

相关推荐
StickToForever32 分钟前
第4章 信息系统架构(五)
经验分享·笔记·学习·职场和发展
黑不溜秋的3 小时前
C++ 设计模式 - 策略模式
c++·设计模式·策略模式
敲敲敲-敲代码4 小时前
【SQL实验】触发器
数据库·笔记·sql
Moonnnn.5 小时前
51单片机学习——动态数码管显示
笔记·嵌入式硬件·学习·51单片机
Dream it possible!5 小时前
LeetCode 热题 100_在排序数组中查找元素的第一个和最后一个位置(65_34_中等_C++)(二分查找)(一次二分查找+挨个搜索;两次二分查找)
c++·算法·leetcode
柠石榴5 小时前
【练习】【回溯No.1】力扣 77. 组合
c++·算法·leetcode·回溯
王老师青少年编程5 小时前
【GESP C++八级考试考点详细解读】
数据结构·c++·算法·gesp·csp·信奥赛
澄澈天空7 小时前
C++ MFC添加RichEditControl控件后,程序启动失败
c++·mfc
电棍2337 小时前
verilog笔记
笔记·fpga开发
让我安静会7 小时前
Obsidian·Copilot 插件配置(让AI根据Obsidian笔记内容进行对话)
人工智能·笔记·copilot