C++数据结构--AVL树

一.什么是AVL树

AVL树,也叫二叉平衡搜索树,就是在BST树的基础上加了平衡操作(平衡的定义是任意节点的左右子树的高度差不超过1)。

其基本框架与BST树相同,不过为了保证其平衡,添加了节点的旋转操作:

cpp 复制代码
template<typename T>
class AVLTree
{
public:
	AVLTree()
		:root(nullptr)
	{ }
	~AVLTree(){}

private:
	struct Node
	{
		Node(T data=T())
			:m_data(data)
			,left(nullptr)
			,right(nullptr)
			,height(1)
		{ }
		


		T m_data;
		Node* left;
		Node* right;
		int height;//节点高度,用于旋转操作
	};
	Node* root;
};
//获取节点高度
int getHeight(Node* node)
{
	return node == nullptr ? 0 : node->height;
}

二.常见操作实现

1.平衡操作

(1).右旋转操作

如果添加一个元素后,左孩子的左子树太高了,那么需要进行右旋转,如图:。

这里的节点node与节点child的孩子节点都发生了变化,它们的高度需要进行更新。

cpp 复制代码
//右旋转操作
Node* rightRotate(Node* node)
{
	Node* child = node->left;
	node->left = child->right;
	child->right = node;

	node->height = max(getHeight(node->left), getHeight(node->right)) + 1;
	child->height = max(getHeight(child->left), getHeight(child->right)) + 1;
	return child;

}

(2).左旋转操作

如果添加一个元素后,右孩子的右子树太高了,那么需要进行左旋转,如图:

这里的节点node与节点child的孩子节点都发生了变化,它们的高度需要进行更新。

cpp 复制代码
//左旋转操作
Node* leftRotate(Node* node)
{
	Node* child = node->right;
	node->right = child->left;
	child->left = node;

	node->height = max(getHeight(node->left), getHeight(node->right)) + 1;
	child->height = max(getHeight(child->left), getHeight(child->right)) + 1;
	return child;

}

(3)左平衡操作

如果添加一个元素后,左孩子的右子树太高了,那么需要进行左平衡,如图:

cpp 复制代码
//左平衡操作(左-右)
Node* leftBalance(Node* node)
{
	node->left= leftRotate(node->left);
	return rightRotate(node);


}

(4)右平衡操作

如果添加一个元素后,右孩子的左子树太高了,那么需要进行右平衡,如图:

cpp 复制代码
//右平衡操作(右-左)
Node* rightBalance(Node* node)
{
	node->right= rightRotate(node->right);

	 return leftRotate(node);


}

2.插入操作

cpp 复制代码
Node* insert(const T& val,Node* node)
{
	if (node == nullptr)
	{
		return new Node(val);

	}
	if (node->m_data == val)
	{
		return node;
	}
	else if (node->m_data > val)
	{
		node->left = insert(val, node->left);
		if ((getHeight(node->left) - getHeight(node->right))>1)
		{
			if (getHeight(node->left->left) >= getHeight(node->left->right))
			{
				node = rightRotate(node);
			}
			else 
			{
				node = rightBalance(node);
			}
		}
		

	}
	else
	{
		node->right = insert(val, node->right);
		if ((getHeight(node->right) - getHeight(node->left)) > 1)
		{
			if (getHeight(node->right->right) >= getHeight(node->right->left))
			{
				node = leftRotate(node);
			}
			else
			{
				node = leftBalance(node);
			}
		}
	}
	node->height = max(getHeight(node->left), getHeight(node->right)) + 1;


	return node;
}

3.删除操作

cpp 复制代码
Node* remove(const T& val, Node* node)
{
	if (node == nullptr)
	{
		return nullptr;
	}
	if (node->m_data > val)
	{
		node->left = remove(val, node->left);
		if ((getHeight(node->right) - getHeight(node->left)) > 1)
		{
			if (getHeight(node->right->right) >= getHeight(node->right->left))
			{
				node = leftRotate(node);
			}
			else
			{
				node = leftBalance(node);
			}
		}
		
	}
	else if(node->m_data < val)
	{
		node->right = remove(val, node->right);
		if ((getHeight(node->left) - getHeight(node->right)) > 1)
		{
			if (getHeight(node->left->left) >= getHeight(node->left->right))
			{
				node = rightRotate(node);
			}
			else
			{
				node = rightBalance(node);
			}
		}
	}
	else
	{
		if (node->left != nullptr && node->right != nullptr)
		{
			if (getHeight(node->left) >= getHeight(node->right))
			{
				Node* pre = node->left;
				while (pre->right != nullptr)
				{
					pre = pre->right;
				}
				node->m_data = pre->m_data;
				
				node->left = remove( pre->m_data, node->left);
			}
			else
			{
				Node* last = node->right;
				while (last->left != nullptr)
				{
					last = last->left;
				}
				node->m_data = last->m_data;
				node->right = remove( last->m_data, node->right);

			}
		}
		else
		{
			if (node->left != nullptr)
			{
				Node* left = node->left;
				delete node;
				return left;
			}
			else if (node->right != nullptr)
			{
				Node* right = node->right;
				delete node;
				return right;
			}
			else
			{
				delete node;
				return nullptr;
			}
		}
	}
	node->height = max(getHeight(node->left), getHeight(node->right)) + 1;

	return node;

}
相关推荐
Cando学算法5 小时前
欧拉回路(一笔画)
数据结构·c++·图论
图码5 小时前
一文搞懂如何判断字符串是否为Pangram(全字母句)
数据结构·算法·网络安全·数字雕刻·ping++
khalil10206 小时前
代码随想录算法训练营Day-43 动态规划10 | 300.最长递增子序列、674. 最长连续递增序列、718. 最长重复子数组
数据结构·c++·算法·leetcode·动态规划·子序列问题
多加点辣也没关系6 小时前
数据结构与算法|第十一章:跳表
数据结构·算法
晚风叙码6 小时前
归并排序:从原理到非递归实现,一文搞定
数据结构·算法
多加点辣也没关系6 小时前
数据结构与算法|第十三章:递归与分治
数据结构·算法
梦梦代码精7 小时前
LikeShop 是否安全可靠?——从架构设计到数据表现的系统性分析
数据结构·团队开发·安全性测试
m0_629494737 小时前
LeetCode 热题 100-----21.搜索二维矩阵 II
数据结构·算法·leetcode
平行侠7 小时前
018二进制GCD(Stein算法)- 用位运算代替除法的最大公因数
数据结构·算法