AVL 树要在插入和删除结点后保持平衡,旋转操作必不可少。关键是理解什么时候应该左旋、右旋和双旋。在Youtube上看到一位老师的视频对这个概念讲解得非常清楚,再结合算法书和网络的博文,记录如下。

1.1 AVL 的旋转

一棵AVL树是其每个节点的左子树和右子树的高度差最多为1的二叉查找树(空树高度定义为-1)。AVL树插入和删除时都可能破坏AVL的特性,可以通过对树进行修正来保证特性,修正方法称为旋转。

下面以4个插入操作为例,说明不同旋转对应的场景。

1.1.1 LL-R

插入结点为6,沿着插入路径向上找到第一个不平衡的结点18;从18开始往下看插入方式,16是18的左孩子的左孩子,这种方式称为LL,用来平衡的旋转方式为右旋,即R。

1.1.2 RR-L

插入结点为48,沿着插入路径向上找到第一个不平衡的结点24;从24开始往下看插入方式,48是24的右孩子的右孩子,这种方式称为RR,用来平衡的旋转方式为左旋,即L。

1.1.3 LR-LR

插入的结点是21,沿着插入路径向上找到第一个不平衡的结点28;从28开始往下看插入方式,21是28的左孩子的右孩子,这种方式称为LR, 用来平衡的旋转方式为双旋转,称为LR旋转。LR旋转即先把R路径行的15进行L旋转,再把L路径上的18进行R旋转。注意顺序。

1.1.4 RL-RL

插入的结点是36,沿着插入路径向上找到第一个不平衡的结点24;从24开始往下看插入方式,36是24的右孩子的左孩子,这种方式称为RL, 用来平衡的旋转方式为双旋转,称为RL旋转。RL旋转即先把L路径行的91进行R旋转,再把R路径上的24进行L旋转。注意顺序。

1.1.5 Problem-Solution

LL问题,使用R旋转;

RR问题,使用L旋转;

LR问题,使用LR旋转(再次强调是先把R分支进行L旋转,再把L分支进行R旋转)

RL问题,使用RL旋转(再次强调是先把L分支进行R旋转,再把R分支进行L旋转)

1.2插入

每次插入后判断树是否平衡,有两种方式:一是每个结点存储平衡因子并随时更新;二是每个结点存储当前的高度(叶结点为0,每向上一层高度+1)。据相关资料,“存储平衡因子所得到的些微的速度优势很难抵消清晰度和相对简明性的损失”,实际需要的也只是两棵子树的高度差,所以建议在每个结点上存储树的高度。

  struct AvlNode
{
	ElementType Element;
	AvlTree Left;
	AvlTree Right;
	int Height;
	struct AvlNode()
	{
		Element = -1;
		Left = NULL;
		Right = NULL;
		Height = 0;
	}
};

插入结点时,如果树的高度不变,那么插入完成;如果高度出现不平衡,则根据插入情况适当做单旋或者双旋,更新高度(并解决好与树其他部分的连接),从而完成插入。只需要判断高度差为2的情况,因为一出现不平衡,就立刻调整,所以高度差不会大于2。整个插入过程,就是二叉搜索树的插入+平衡调整。

AvlTree Insert(ElementType x, AvlTree T)
{
	if (T == NULL)
	{
		//创建并返回一棵单结点的树
		T = new AvlNode();
		assert(T);
		T->Element = x;
		return T;
	}
	else if (x < T->Element)
	{
		T->Left = Insert(x, T->Left);
		//调整最深的不平衡的结点
		//高度不平衡时,该结点的两棵子树高度差为2
		if (Height(T->Left) - Height(T->Right) == 2)
		{
			if (x < T->Left->Element)//Problem: LL->Solution:R
				T = SingleRotateR(T);
			else  //Problem: LR->Solution: LR
				T = DoubleRotateLR(T);
		}
	}
	else if (x > T->Element)
	{
		T->Right = Insert(x, T->Right);
		if (Height(T->Right) - Height(T->Left) == 2)
		{
			if (x > T->Right->Element)//Problem: RR->Solution:L
				T = SingleRotateL(T);
			else//Problem: RL->Solution:RL
				T = DoubleRotateRL(T);
		}
	}
	T->Height = Max(Height(T->Left), Height(T->Right)) + 1;
	return T;
}

下面来看看旋转。沿插入点向上的第一个不平衡的点(也是最深的不平衡点),记为α点,则L旋转和R旋转,LR旋转和RL旋转,分别都是关于这一点的镜像对称,因此我们就讨论R旋转和LR旋转。

R旋转:

在子树X中插入结点,假设K2为α点,这是LL问题,所以要进行R旋转。根据上图调整指针:

Position SingleRotateR(Position k2)
{
	Position k1;

	k1 = k2->Left;
	k2->Left = k1->Right;
	k1->Right = k2;

	k2->Height = Max(Height(k2->Left), Height(k2->Right)) + 1;
	k1->Height = Max(Height(k1->Left), Height(k1->Right)) + 1;

	//k1为新的根结点
	return k1;
}

LR旋转:

如上图,假设K2是α点,插入位置是在Y子树中。这里我们把Y子树看做一个由一个根结点和两棵子树构成,如下图:

根据我们在第一节中讲的,LR问题用LR旋转解决,先对R分支进行L旋转,再对L分支进行R旋转:

Position DoubleRotateLR(Position k3)
{
	k3->Left = SingleRotateL(k3->Left);
	return SingleRotateR(k3);
}

1.3 删除

AVL的删除和二叉搜索树的删除很像,多了平衡的操作。 这里贴出伪代码来理解,完整代码实现:https://github.com/yulifromchina/exercise/tree/master/algorithm/tree/avl%20tree。

伪代码大致如下:

Delete(Tree, data)
{
	if(data < Tree->data) Delete(Tree->left, data);balance(Tree);
        else if(data > Tree->data) Delete(Tree->right,data);balance(Tree);
        else
             if(Tree->left && Tree->right)
                    replace Tree->data with successor or precursor;
                    Delete(Tree->right/left, successor/precursor);
                    balance(Tree);
             else
                   if(Tree->left==NULL)
                         Tree = Tree->right;
                   else if(Tree->right==NULL)
                         Tree = Tree->left;
                   free(Tree's origin memory)
        update(Tree's Height)
}

当待删除结点是叶子结点时,直接删除即可,不用考虑平衡,因为这是递归调用过程,返回上一层叶子结点的父结点时会进行平衡;当待删除结点是只有左子树或右子树时,用该结点的父节点指向左子树或者右子树,再释放该结点;当待删除结点儿女双全时,用前驱或后继赋值,并递归删除前驱或后继。是不是很二叉搜索树很像?另外,每次递归回来都需要平衡一下,这其实就是从插入的结点向上平衡到根结点的过程。

1.4 参考

  1. Youtebu视频:Basics of AVL tree.
  2. 《数据结构与算法分析:C语言描述》

  3. 一步一步写平衡二叉树(AVL树)

(完)

AVL树的插入与删除的更多相关文章

  1. AVL 树的插入、删除、旋转归纳

    参考链接: http://blog.csdn.net/gabriel1026/article/details/6311339   1126号注:先前有一个概念搞混了: 节点的深度 Depth 是指从根 ...

  2. AVL树的插入和删除

    一.AVL 树 在计算机科学中,AVL树是最早被发明的自平衡二叉查找树.在AVL树中,任一节点对应的两棵子树的最大高度差为 1,因此它也被称为高度平衡树.查找.插入和删除在平均和最坏情况下的时间复杂度 ...

  3. 创建AVL树,插入,删除,输出Kth Min

    https://github.com/TouwaErioH/subjects/tree/master/C%2B%2B/PA2 没有考虑重复键,可以在结构体内加一个int times. 没有考虑删除不存 ...

  4. AVL树的插入操作(旋转)图解

    =================================================================== AVL树的概念       在说AVL树的概念之前,我们需要清楚 ...

  5. B树和B+树的插入、删除图文详解

    简介:本文主要介绍了B树和B+树的插入.删除操作.写这篇博客的目的是发现没有相关博客以举例的方式详细介绍B+树的相关操作,由于自身对某些细节也感到很迷惑,通过查阅相关资料,对B+树的操作有所顿悟,写下 ...

  6. B树和B+树的插入、删除图文详解(good)

    B树和B+树的插入.删除图文详解 1. B树 1. B树的定义 B树也称B-树,它是一颗多路平衡查找树.我们描述一颗B树时需要指定它的阶数,阶数表示了一个结点最多有多少个孩子结点,一般用字母m表示阶数 ...

  7. AVL树的插入删除查找算法实现和分析-1

    至于什么是AVL树和AVL树的一些概念问题在这里就不多说了,下面是我写的代码,里面的注释非常详细地说明了实现的思想和方法. 因为在操作时真正需要的是子树高度的差,所以这里采用-1,0,1来表示左子树和 ...

  8. B+树的插入、删除(附源代码)

    B+ Tree Index B+树的插入 B+树的删除 完整测试代码 Basic B+树和B树类似(有关B树:http://www.cnblogs.com/YuNanlong/p/6354029.ht ...

  9. MySQL B+树 的插入与删除

    一.MySQL Index 的插入 有如下B+树,其高度为2,每页可存放4条记录,扇出为5.所有记录都在叶子节点上, 并且是顺序存放,如果用户从最左边的叶子节点开始顺序遍历,可以得到所有简直的顺序 排 ...

随机推荐

  1. POJ 3984 路径输出

    迷宫问题 Description 定义一个二维数组: int maze[5][5] = { 0, 1, 0, 0, 0, 0, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 1, 1, ...

  2. Java之数字处理类浅析

    包装类: 数据类型相对的包装类:byte---Byteshort---Shortint---Integerlong---Long float---Floatdouble---Double boolea ...

  3. LoadRunner接口测试Error -27225报错解决

    今天依照规范写了一个接口测试脚本,再执行的时候报Error -27225,核对了接口字段和字段值没发现错误,百度搜Error -27225错误没有相关解释.这个问题经过溯源找到了问题的所在,为了互帮互 ...

  4. ZooKeeper快速学习

    "一入Java深似海",过去自身对于分布式的接触,始终处于使用别人构建的框架的水平,最多就是在nginx配置一下第4层的负载均衡(最后有介绍).随着java使用深入,本文将重点理解 ...

  5. Apache Flume 1.7.0 各个模块简介

    Flume简介 Apache Flume是一个分布式.可靠.高可用的日志收集系统,支持各种各样的数据来源,如http,log文件,jms,监听端口数据等等,能将这些数据源的海量日志数据进行高效收集.聚 ...

  6. 存储结构比较vector,list,dequeue,stack(转)

        vector适用:对象数量变化少,简单对象,随机访问元素频繁list适用:对象数量变化大,对象复杂,插入和删除频繁最大的区别是,list是双向的,而vector是单向的.因此在实际使用时,如何 ...

  7. strtok函数 分类: c++ 2014-11-02 15:24 214人阅读 评论(0) 收藏

    strtok函数是cstring文件中的函数 strtok函数是cstring文件中的函数 其功能是截断字符串 原型为:char *strtok(char s[],const char *delin) ...

  8. spring boot 整合mybatis + swagger2

    之前使用springMVC+spring+mybatis,总是被一些繁琐的xml配置,有时候如果配置出错,还要检查各种xml配置,偶然接触到了spring boot 后发现搭建一个web项目真的是1分 ...

  9. ASP.NET MVC 开发微信支付H5(外置浏览器支付)

    H5支付是指商户在微信客户端外的移动端网页展示商品或服务,用户在前述页面确认使用微信支付时,商户发起本服务呼起微信客户端进行支付. 主要用于触屏版的手机浏览器请求微信支付的场景.可以方便的从外部浏览器 ...

  10. hexo博客MathJax公式渲染问题

    这个问题自己很早以前便碰到了,用MathJax语法写的一些公式,在本地Markdown编译器上渲染是没问题的,可是部署到hexo博客中就出现问题了,之前我是使用图片代替公式应付过去了,今天从网上找了一 ...