实现Avl平衡树

 

一、介绍

  AVL树是一种自平衡的二叉搜索树,它由Adelson-Velskii和 Landis于1962年发表在论文《An algorithm for the organization of information》中。AVL树的特点是,其左右子树的高度差的绝对值小于2(空树的高度定义为 -1,无子树的树高度为0)。如下图所示,左边的二叉树为AVL树,而右边的二叉树root节点的左子树高度为2,右子树高度为0,高度差为2,不是AVL树。与普通二叉树相同的是查找和遍历;但是插入和删除操作可能会破坏AVL的平衡结构,这也是实现AVL树的难点所在。

二、定义

  AVL的特点是平衡,每个节点平衡依赖于其左右子树的高度,因此,每个树节点都有一个height值。这里的定义和实现来自于《数据结构和算法分析-C语言描述》一书,具体定义如下:

/*
filename: Avltree.h
from: chapter 4 tree, page 87, Data Structure and Algorithm Analysis
date: 2013-08-06
function: declearation of AVL Tree
*/
#ifndef _AVLTREE_H_
#define _AVLTREE_H_ // 定义树节点
struct AvlNode{
double element; //内容元素
AvlTree left; //左孩子
AvlTree right; //右孩子
int height; //高度,无子节点时为 0,NULL时为 -1
};
typedef struct AvlNode * Position;
typedef struct AvlNode * AvlTree; void InitAvl(AvlTree *T);
AvlTree MakeEmpty(AvlTree T);
Position Find(double X, AvlTree T);
Position FindMin(AvlTree T);
Position FindMax(AvlTree T);
AvlTree Insert(double X, AvlTree T);
AvlTree Delete(double X, AvlTree T);
double Retrieve(Position P); #endif

  对节点高度的控制是通过Height(Position P)实现的,很重要的一点是它规定了空树的高度(此时调用P->height会报错)。Height(Position P)实现如下:

static int Height(Position P){
if(P==NULL) return -1;
else return P->height;
}

三、初始化和置空树

  创建一个新树时,使用InitAvl()函数将其初始化为空。删除所有节点使用MakeEmpty()函数,代码如下:

/*
filename: Avltree.c
*/
void InitAvl(AvlTree *T){
*T = NULL;
}
AvlTree MakeEmpty(AvlTree T){
if(T!=NULL){
MakeEmpty(T->left);
MakeEmpty(T->right);
free(T);
}
return NULL;
}

四、实现查找

  AVL树的查找与普通二叉树没有区别,可以通过递归或者非递归的方式实现,都比较简单。这里使用的是递归,需要注意,Find()、FindMin()和FindMax()方法前的return 比不可少,否则递归得出的结果无法向上传递。

Position Find(double X, AvlTree T){
if(T!=NULL){
if(X==T->element) return T;
else if(X>T->element) return Find(X, T->right);
else return Find(X, T->left);
}else return T;
}
Position FindMin(AvlTree T){
if(T!=NULL && T->left!=NULL) return FindMin(T->left);
else return T;
}
Position FindMax(AvlTree T){
if(T==NULL ||T->right==NULL) return T;
else return FindMin(T->right);
}

五、实现插入

  在插入之前,AVL树中每个节点都是平衡的,插入元素以后,可能会导致 插入节点到根节点路径上的节点的平衡发生改变,而插入节点的子节点的平衡不变。沿着这条路径上行,对路径上的每个节点,检查其平衡状态,修正不平衡的节点,修正height,一直检查和修正到root节点。最后可以得到的是一棵平衡的AVL树。

  假设我们遍历到的节点是a,如果a满足平衡条件,则上行一个节点。如果不满足,则有四种情况:

  1. 插入到a的左儿子的左子树后,平衡被打破,(结果:左高右低,left-left型)(例子见上图)
  2. 插入到a的左儿子的右子树后,平衡被打破,(结果:左高右低,left-right型)
  3. 插入到a的右儿子的左子树后,平衡被打破,(结果:右高左低,right-left型)
  4. 插入到a的右儿子的右子树后,平衡被打破,(结果:右高左低,right-right型)

情形1和情形4对称,情形2和情形3对称。这里先看情形1。需要通过旋转来调整这几种不平衡状态

1、Left-left case

  这里,K2是当前要处理的节点。可以看到K2->left 比 K2->right 要高两层,而X比Y要高一层。而插入之前,k2满足平衡条件。插入新元素导致 X 长出一层,使其比Z高两层。重新调整平衡的方法是使X上移一层,Z下移一层,同时将K1设置为当前节点。

实现如下:

/*
filename: AvlTree.c
*/
static Position SingleRotationWithLeft(Position K2){ //左孩子的Height太大
Position 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(K2))+1;
return K1;
}

2、right-right case

  这种情况和与前一种没有本质差别,这里仅仅附上图示和代码

// filename: Avltree.c
static Position SingleRotationWithRight(Position K2){ //右孩子的Height太大,且大的分支在其右孩子
Position K1 = K2->right;
K2->right = K1->left;
K1->left = K2;
K2->height = Max(Height(K2->left), Height(K2->right)) + 1;
K1->height = Max(Height(K1->right), K2->height)+1;
return K1;
}

3、left-right case

这种情况下,使用单次旋转后,可以看出当前节点(K1)仍然是不平衡的,因此我们需要进行两次旋转。旋转方法见图

等价于

  

  这里使用了两次单旋转,先旋转K1和K2,再把K2置为当前节点。(通过图可以直观地看出K1->element < K2->element < K3->element)

// filename: Avltree.c
static Position DoubleRotationWithLeft(Position K2){ // 左孩子太大,且大的分支在左孩子的右孩子上
K2->left = SingleRotationWithRight(K2->left);
K2 = SingleRotationWithLeft(K2); return K2;
}

4、right-left case

  这种情形与情形3类似,不多解释,代码如下

// filename: Avltree.c
static Position DoubleRotationWithRight(Position K2){ //右孩子太大,且大的分支是右孩子的左孩子
K2->right = SingleRotationWithLeft(K2->right);
K2 = SingleRotationWithRight(K2);
return K2;
}

  这四种旋转实现了平衡AVL树的功能,下面就插入元素分情况进行讨论。

  如果T为空树,则插入比较简单,直接分配内存,并存储即可。如果T不为空树,则需要分情况讨论。这里再强调一次,

  “在插入之前,AVL树中每个节点都是平衡的,插入元素以后,可能会导致 插入节点到根节点路径上的节点的平衡发生改变,而插入节点的子节点的平衡不变。沿着这条路径上行,对路径上的每个节点,检查其平衡状态,修正不平衡的节点,修正height,一直检查和修正到root节点。最后可以得到的是一棵平衡的AVL树。”

  每个节点的平衡状态是由其左右子节点的高度差决定的,使用递归可以保证更新完子节点以后,上溯到父节点进行检查更新。

// filename: Avltree.c
AvlTree Insert(double X, AvlTree T){
if(T==NULL){ // T 为空树
T = (AvlTree)malloc(sizeof(struct AvlNode));
if(T==NULL){
fprintf(stderr, "no space available!\n");
abort();
}else{
T->element = X;
T->left = T->right = NULL;
T->height = 0;
}
}else if(X<T->element){
T->left = Insert(X, T->left);
if(Height(T->left) - Height(T->right)==2){
if(X<T->left->element) // X被插入到其左孩子的左子树中,left-left case
T = SingleRotationWithLeft(T);
else // X被插入到其左孩子的右子树中,left-right case
T = DoubleRotationWithLeft(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) // X被插入到右孩子的左子树中, right-left case
T = DoubleRotationWithRight(T);
else // X 被插入到其右孩子的右子树中, right-right case
T = SingleRotationWithRight(T);
}
}
// else if x== T->left->element, no need to know T->height = Max(Height(T->left),Height(T->right)) + 1; // 更新节点的高度 Max()为辅助函数
return T;
}

六、实现删除

  节点的删除比Insert麻烦一些,我们分情形处理。

  1. T为空树
  2. 要删除的元素不存在
  3. 要删除的元素位于叶子节点上
  4. 要删除的元素位于非叶子节点上

  情形1 和情形2不需要太多处理,直接返回T;情形3则需要递归删除,并逐级上溯纠正树的平衡性;情形4比较复杂,这里将详细讨论。

  假设要删除的节点为 a,如果a->left 或者 a->right为空树,则将 a = a->right(假设a->left==NULL), 释放原来a节点的空间,即可。然后逐个上溯到父节点,并调整其平衡性。如果 a的左右子树均不为空,则用其右子树的最小值取代 a中的值(a->element = FindMin(a->right)),递归删除右子树的最小值,然后调整a的平衡。最后逐个上溯到父节点,并调整其父节点的平衡性。

        if(!T->left){  // 第一种情况: 左子树为NULL, 删除后需要重新计算高度(减一),但不需要调节平衡
pos = T;
T = T->right;
free(pos);
}else if(!T->right){ //第二种情况:右子树为NULL,删除后需要重新计算高度(减一),但不需要调节平衡
pos = T;
T = T->left;
free(pos);
}else{ // 第三种情况,左右子树均不为空
// 用右子树的最小值代替X,删除右子树最小值,并调节其平衡
pos = FindMin(T->right);
T->element = pos->element;
Delete(pos->element, T->right); // 递归删除最小值,计算高度,并调节平衡至 T->right
// 但无法保证 T 的平衡,因此需要调节
if(Height(T->left) - Height(T->right)==2){
if(Height(T->left->right)-Height(T->left->left)==1) // left-right型
T = DoubleRotationWithLeft(T);
else T = SingleRotationWithLeft(T); // left-left型
}
}

  Delete的全部实现如下:

AvlTree Delete(double X, AvlTree T){
Position pos;
if(!T) return NULL; // when T ==NULL // 递归删除
if(X<T->element){
T->left = Delete(X, T->left);// 删除后,左子树元素个数减少,可能为 NULL
if(Height(T->right)-Height(T->left)==2){ // 平衡被打破,右子树高度过高
if(T->left==NULL){
// need to identify them
if(T->right->right==NULL) T = DoubleRotationWithRight(T);
else T = SingleRotationWithRight(T);
}else if(X<T->left->element) // X小于左子树的元素,为 left-right型
T = DoubleRotationWithLeft(T);
else // X 大于左子树的元素,左子树的右子树的元素被删除,为left-left型
T = SingleRotationWithLeft(T);
}
}else if(X>T->element){
T->right = Delete(X, T->right); //删除后,右子树的元素个数减少, 可能为 NULL
if(Height(T->left)-Height(T->right)==2){ //平衡被打破,左子树高度过高
if(T->right==NULL){
//left-left or left-right
if(T->left->left==NULL) T = DoubleRotationWithLeft(T);
else T = SingleRotationWithLeft(T);
}else if(X<T->right->element) //X小于右子树元素,,为right-right型
T = SingleRotationWithRight(T);
else //X 大于右子树元素,为right-left型
T = DoubleRotationWithRight(T);
}
}else{ // X == T->element if(!T->left){ // 第一种情况: 左子树为NULL, 删除后需要重新计算高度(减一),但不需要调节平衡
pos = T;
T = T->right;
free(pos);
}else if(!T->right){ //第二种情况:右子树为NULL,删除后需要重新计算高度(减一),但不需要调节平衡
pos = T;
T = T->left;
free(pos);
}else{ // 第三种情况,左右子树均不为空
// 用右子树的最小值代替X,删除右子树最小值,并调节其平衡
pos = FindMin(T->right);
T->element = pos->element;
Delete(pos->element, T->right); // 递归删除最小值,计算高度,并调节平衡至 T->right
// 但无法保证 T 的平衡,因此需要调节
if(Height(T->left) - Height(T->right)==2){
if(Height(T->left->right)-Height(T->left->left)==1) // left-right型
T = DoubleRotationWithLeft(T);
else T = SingleRotationWithLeft(T); // left-left型
}
}
}
// things does not work when T ==NULL
if(T) T->height = Max(Height(T->left),Height(T->right)) + 1;
return T;
}

参考引用: 《数据结构与算法分析--C语言描述》

实现Avl平衡树的更多相关文章

  1. Python与数据结构[3] -> 树/Tree[2] -> AVL 平衡树和树旋转的 Python 实现

    AVL 平衡树和树旋转 目录 AVL平衡二叉树 树旋转 代码实现 1 AVL平衡二叉树 AVL(Adelson-Velskii & Landis)树是一种带有平衡条件的二叉树,一棵AVL树其实 ...

  2. 数据结构学习-AVL平衡树

    环境:C++ 11 + win10 IDE:Clion 2018.3 AVL平衡树是在BST二叉查找树的基础上添加了平衡机制. 我们把平衡的BST认为是任一节点的左子树和右子树的高度差为-1,0,1中 ...

  3. AVL平衡树的插入例程

    /* **AVL平衡树插入例程 **2014-5-30 11:44:50 */ avlTree insert(elementType X, avlTree T){ if(T == NULL){ T = ...

  4. AVL 平衡树

    AVL是一种平衡二叉树,它通过对二叉搜索树中的节点进行旋转使得二叉搜索树达到平衡.AVL在所有的平衡二叉搜索树中具有最高的平衡性. 定义 平衡二叉树或者为空树或者为满足如下性质的二叉搜索树: 左右子树 ...

  5. AVL平衡树(非指针实现)

    看了网上三四篇博客,学习了AVL树维护平衡的方式.但感觉他们给出的代码都有一点瑕疵或者遗漏,懂得了思想之后,花了一些时间把他们几篇的长处结合起来,没有使用指针,实现了一下.每个小逻辑功能都抽象成了函数 ...

  6. 数据结构——AVL平衡树

    1.是二叉搜索树(Binary Search Tree) 2.树和所有左右子树高度之差为-1,0,1 平衡因子(balance factor) =右子树高度-左子树高度 平衡化旋转: 1.从插入位置向 ...

  7. My implementation of AVL tree

    C++实现的avl平衡树 #include <stdlib.h> #include <time.h> #include <string.h> #include &l ...

  8. [SinGuLaRiTy] 平衡树

    [SinGuLaRiTy-1009] Copyright (c) SinGuLaRiTy 2017. All Rights Reserved. 二叉查找树 二叉查找树是指具有下列性质的非空二叉树: ⑴ ...

  9. 数据结构--Avl树的创建,插入的递归版本和非递归版本,删除等操作

    AVL树本质上还是一棵二叉搜索树,它的特点是: 1.本身首先是一棵二叉搜索树.   2.带有平衡条件:每个结点的左右子树的高度之差的绝对值最多为1(空树的高度为-1).   也就是说,AVL树,本质上 ...

随机推荐

  1. ar命令提取.a时刻,一个错误 is a fat file (use libtool(1) or lipo(1) and ar(1) on it)

    在减压.a当文件,据报一个类别似 xxx.a is a fat file (use libtool(1) or lipo(1) and ar(1) on it)的错误,经过查找资料,原来是由于该.a文 ...

  2. C# 打开指定文件或网址

    System.Diagnostics.Process.Start的妙用: 文件夹打开时自动选中一个文件,比如自动选中此目录下的指定文件方法: Process.Start("Explorer& ...

  3. Zepto

    移动开发流量省起来之Zepto 事情是这样的:最近开发的一个手机网站客户反应访问起来特别慢,刷了半天才能看到页面,然后问我们是不是网站出问题了.于是我赶紧找了各种手机测试一下,没有问题,首先排除程序错 ...

  4. JavaScript中的单引号和双引号报错的解决方法

    在使用JavaScript显示消息或者传递字符数据的时候,经常会碰到数据中夹杂单引号(')或者双引号("),这种语句往往会造成JavaScript报错.对此一般采用/'或者/"的解 ...

  5. checkbox 选择功能和反选

    使用jQuery实现checkbox全补选和反选功能.什么时候checkbox选择禁用时,不涉及功能 <!DOCTYPE html> <html> <head> & ...

  6. yii性能调节

    网络应用程序的性能受很多因素的影响.数据库存取,文件系统操作,网络带宽等都是潜在的影响因素. Yii 已在各个方面减少框架带来的性能影响.但是在用户的应用中仍有很多地方可以被改善来提高性能. 1. 开 ...

  7. Nancy和MVC的简单对比

    Nancy和MVC的简单对比 在上一篇的.NET轻量级MVC框架:Nancy入门教程(一)——初识Nancy中,简单介绍了Nancy,并写了一个Hello,world.看到大家的评论,都在问Nancy ...

  8. leetcode第31题--Longest Valid Parentheses

    Given a string containing just the characters '(' and ')', find the length of the longest valid (wel ...

  9. ser2net使用

    在ubuntu下或者openwrt下安装了ser2net程序之后,可以将串口中的数据转发为以太网数据. 设置在/etc/ser2net.conf中最后: 3002:0:/dv/ttyUSB0:1152 ...

  10. 批处理中set截取字符具体解释

    set截取字符具体解释  在批处理中,set的功能有点繁杂:设置变量.显示环境变量的名及值.做算术运算.等待用户的输入.字符串截取.替换字符串,是我们经常使用的命令之中的一个. 在字符串截取方面,新手 ...