数据结构--AVL树的插入和四种自旋方式

参考原视频:https://www.icourse163.org/learn/ZJU-93001?tid=1003013004#/learn/content?type=detail&id=1004242207

AVL树,又名平衡二叉树,是一种计算机科学中最先发明的自平衡二叉树。

emmm,要知道什么是平衡二叉树,先说说二叉搜索树吧,

那二叉搜索树是什么呢?

我们先定义一个树节点:

struct node{

int data;

    node *left;

    node *right;

};

先看图1说明,当把5作为中心节点时,2是5的左分叉,那么2小于5;同时8是5的右分叉,而8大于5,所有的中心节点和左右分叉的关系都是这样的。

这就是所谓的二叉搜索树。

而AVL树就是二叉树的升级,可以看到右图是一棵普通的二叉树,树的左右分叉不是很平衡,左边密集右边稀疏,这样会导致搜索的效率降低,所有就诞生了AVL树!


图1

AVL树为什么叫AVL呢,因为它的发明者的名字叫Adelson Velsky和E. M. Landis。

那么让我们着重来探讨以下AVL树是如何实现自平衡的吧。

原理很简单:

每次插入一个新的节点时,如果存在某个节点的平衡被破坏(左子树与右子树的树高之差大于1),那么就检测被破坏时,插入点(麻烦制造者)的位置与该节点(被破坏者)的位置关系,进行相应的自选方式。

因此,AVL树的节点需要新加一个树高参数。

故新的node如下:

struct node{   

        int data;

        node *left;

        node *right;

        int height;   

node(){};

node(int x):data(x),left(NULL),right(NULL),height(1){}

};

typedef  node* AVLtree 真正定义了AVL树

如何更新树高呢?

int GetHeight(AVLtree a){

if(a==NULL)return 0;

    a->height=max(GetHeight(a->left),GetHeight(a->right))+1;

    return a->height;

}

如果在外面使用

GetHeight(t)

就可以把AVL树 t 的所有节点的树高都更新一遍。

所以,如何检验节点(node *)t是否平衡呢?

if(abs(GetHeight(t->left)-GetHeight(t->right))>1)

只需这么一个简单的if条件判断即可。

现在我们来到了最难的部分,自旋方式吧。

先说说LL旋转(左单旋),

条件:麻烦制造者位于被破坏者的左子树的左分叉部分


图2

如图2所示,那么我们所要做的就是把三者的中间节点70提起来,让70成为老大!

原来的88变成了70的右儿子了。

让我们看一看更复杂的情况

图3

如图3所示,我们将新节点插入于BL部分,导致A被破坏了,所以我们进行了LL旋转,可以看到,当我们对A进行LL旋转时,其实是将BL----B-----A进行了向右的拉扯,或者更像是顺时针的旋转,那这个旋转的名字却叫左单旋!左是根据麻烦制造者BL相对A的位置命名的!

那么如何用代码实现呢?

AVLtree L_rotation(AVLtree t){

AVLtree root=t->left;

    t->left=root->right;

    root->right=t;

    return root;

}

RR旋转(右单旋)与左单旋十分类似,只不过麻烦制造者位于被破坏者右子树的右分叉部分罢了。代码就不给大家了,给两张图感受感受

图4


图5

下面来说说更复杂的旋转

RL旋转,

条件:麻烦制作者位于被破坏者的右子树的左分叉部分


图6

如图6,插入了90之后,70的平衡被破坏了,所以让我们聚焦70,96,88这三点(被破坏者,被破坏者的右子树的根节点,右子树的左子树的根节点),需要把三个节点的中间值,也就是88提起来做老大,如何70做88的左儿子,96做右儿子。而88原来的左子树,成了70的右子树,88原来的右子树,成了96的左子树。


如果用代码实现,则更简单了(前提是已经实现了LL旋转和RR旋转)


如果是RL旋转某一棵树t,只需先对t的右子树进行一次LL旋转,再对t进行一次RR旋转,大家自己画个草图看看是不是这么一回事!

所以两行代码就好了

AVLtree RL_rotation(AVLtree t){

t->right=L_rotation(t->right);

    return R_rotation(t);

}

好了,讲完了RL旋转,LR旋转也就都会了吧(大概吧。。。

所以,我们可以看到,AVL树在插入的过程会不断地旋转,导致其根节点不停改变???好像还没讲插入,下面给出插入的代码,根据插入的代码,自行写出删除的代码,完结!!

AVLtree insert(AVLtree t,int x){

        if(t==NULL){ t=new node(x); return t; }

        if(x<t->data){//左树

                    t->left=insert(t->left,x);

                    GetHeight(t);

                   if(GetHeight(t->left)-GetHeight(t->right)>1){//not balance

                                    if(x<t->left->data){ t=L_rotation(t); }

                                     else if(x>t->left->data){ t=LR_rotation(t); }

                  }

         } else if(x>t->data){

                    t->right=insert(t->right,x);

                    GetHeight(t);

                   if(GetHeight(t->right)-GetHeight(t->left)>1){

                                if(x>t->right->data){ t=R_rotation(t); }

                                else if (x<t->right->data){ t=RL_rotation(t); }

                    }

     }

      return t;

}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,047评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,807评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,501评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,839评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,951评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,117评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,188评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,929评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,372评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,679评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,837评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,536评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,168评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,886评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,129评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,665评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,739评论 2 351

推荐阅读更多精彩内容