算法日记(七)之红黑树(基础版)

今天,我们来看看二叉树中的一种特殊结构——红黑树。网上也都流传说红黑树在平时程序员面试中经常出现,且会出现得比较难。虽然红黑树在《数据结构与算法》这本书中没有提高,但还是有必要学好红黑树。学红黑树以前,我们有必要来复习二叉排序树的内容(博主前面的文章有一期是关于二叉排序树的)。主要本期文章只是分享红黑树的基础内容,后期还有关于红黑树的进阶版,尽请期待吧。

1.复习二叉排序树

二叉查找树(BST)具备什么特性呢?

1.子树上所有结点的值均小于或等于它的根结点的值。

2.子树上所有结点的值均大于或等于它的根结点的值。

3.左、右子树也分别为二叉排序树。

2.红黑树:

当我们向二叉排序树添加结点时,难免有时候会出现二叉树出现瘸子的情况,为了解决二叉排序树多次插入新节点出现的不平衡这种情况。就有了今天的主角——红黑树。红黑树是一种自平衡的二叉查找树。

红黑树除了有二叉查找(排序)树的性质外,还有以下的性质:

1.结点是红色或黑色。

2.根结点是黑色。

3.每个叶子结点都是黑色的空结点(NIL结点)。

4 每个红色结点的两个子结点都是黑色。(从每个叶子到根的所有路径上不能有两个连续的红色结点)

5.从任一结点到其每个叶子的所有路径都包含相同数目的黑色结点。

红黑树从根到叶子的最长路径不会超过最短路径的两倍,但插入结点时,什么情况下会破坏红黑树的规则,什么情况下不会破坏规则呢? 

根据上图,我们来分析以下吧

1.向原红黑树插入值为14的新结点:由于父结点15是黑色结点,因此这种情况并不会破坏红黑树的规则,无需做任何调整。

2.向原红黑树插入值为21的新结点:由于父结点22是红色结点,因此这种情况打破了红黑树的规则4(每个红色结点的两个子结点都是黑色),必须进行调整,使之重新符合红黑树的规则。

3.红黑树调整的方法

变色和旋转(左旋转和右旋转)

一.变色

为了重新符合红黑树的规则,尝试把红色结点变为黑色,或者把黑色结点变为红色。但是,仅仅把一个结点变色,会导致相关路径凭空多出一个黑色结点,这样就打破了规则5。因此,我们需要对其他结点做进一步的调整,

二.左旋转

逆时针旋转红黑树的两个结点,使得父结点被自己的右孩子取代,而自己成为自己的左孩子。说起来很怪异,大家看下图:

三.右旋转:

顺时针旋转红黑树的两个结点,使得父结点被自己的左孩子取代,而自己成为自己的右孩子。大家看下图:

 想要调整时,那什么用变色,什么时候用旋转呢。主要分以下几种情况,一起来看看吧

1.新结点(A)位于树根,没有父结点。

这种局面,直接让新结点变色为黑色,规则2得到满足。同时,黑色的根结点使得每条路径上的黑色结点数目都增加了1,所以并没有打破规则5。

2.新结点(B)的父结点是黑色

这种局面,新插入的红色结点B并没有打破红黑树的规则,所以不需要做任何调整。

 

3.新结点(D)的父结点和叔叔结点都是红色 

这种局面,两个红色结点B和D连续,违反了规则4。因此我们先让结点B变为黑色:

 这样一来,结点B所在路径凭空多了一个黑色结点,打破了规则5。因此我们让结点A变为红色:

 

这时候,结点A和C又成为了连续的红色结点,我们再让结点C变为黑色: 

 

 4.新结点(D)的父结点是红色,叔叔结点是黑色或者没有叔叔,且新结点是父结点的右孩子,父结点(B)是祖父结点的左孩子。

 我们以结点B为轴,做一次左旋转,使得新结点D成为父结点,原来的父结点B成为D的左孩子:这样一来,进入了下面第5点。

 5.新结点(D)的父结点是红色,叔叔结点是黑色或者没有叔叔,且新结点是父结点的左孩子,父结点(B)是祖父结点的左孩子。

我们以结点A为轴,做一次右旋转,使得结点B成为祖父结点,结点A成为结点B的右孩子: 我们让结点B变为黑色,结点A变为红色

 

 以上就是红黑树插入操作所涉及的5种局面。

下面我们来看一看添加一个结点21,应该怎么调整吧

显然,新结点21和它的父结点22是连续的红色结点,违背了规则4,我们应该如何调整呢?

让我们回顾一下刚才讲的5种局面,当前的情况符合局面3:

“新结点的父结点和叔叔结点都是红色。”

于是我们经过三次变色,22变为黑色,25变为红色,27变为黑色:

经过上面的调整,以结点25为根的子树符合了红黑树规则,但结点25和结点17成为了连续的红色结点,违背规则4。

于是,我们把结点25看做一个新结点,正好符合局面5的镜像:

“新结点的父结点是红色,叔叔结点是黑色或者没有叔叔,且新结点是父结点的右孩子,父结点是祖父结点的右孩子”

于是我们以根结点13为轴进行左旋转,使得结点17成为了新的根结点:
接下来,让结点17变为黑色,结点13变为红色:

好啦,关于红黑树的基础知识就分享到这啦,当然红黑树没那么简单,后面还有插入删除结点操作,等博主理解弄懂后,再做一期文章跟大家分享吧

本贴为博主亲手整理。如有错误,请评论区指出,一起进步。谢谢大家的浏览.

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值