数据结构与算法之红黑树
1、引言
在二叉树中已经探讨过,如果按照随机顺序插入树节点,绝大多数都会出现不平衡的情况。最坏的情况,插入的数据时有序的,二叉树将会变成链表,插入、删除的效率将会严重地降低。
下图就是按照数据升序的顺序插入二叉树的情况:
文章图片
1.png 红黑树就是一种解决非平衡树的方法,它是增加了某些特点的二叉搜索树。
为了能较快的时间来搜索一颗树,需要保证树总是平衡的(或者至少大部分是平衡的),就是说对树中的每个节点,它左边的后代数量和它右边的后代数量应该大致相等。
2、红黑规则
当插入(或者删除)一个节点时,必须遵循一定的规则,它们被称为红黑规则。如果遵循这些规则,树就是平衡的:
- 每一个节点不是黑色就是红色
- 根总是黑色的
- 如果节点是红色的,则它的子节点必须是黑色的,反之则不一定成立
- 从根到叶节点或空子节点的每条路径,必须包含相同数目的黑色节点
从根节点到叶节点路径上的黑色节点的数目被称为黑色高度。
【数据结构与算法之红黑树】这些约束强制了红黑树的关键性质: 从根到叶子的最长的可能路径不多于最短的可能路径的两倍长。结果是这个树大致上是平衡的。因为操作比如插入、删除和查找某个值的最坏情况时间都要求与树的高度成比例,这个在高度上的理论上限允许红黑树在最坏情况下都是高效的,而不同于普通的二叉查找树。
要知道为什么这些特性确保了这个结果,注意到规则4导致了路径不能有两个毗连的红色节点就足够了。最短的可能路径都是黑色节点,最长的可能路径有交替的红色和黑色节点。因为根据规则4所有最长的路径都有相同数目的黑色节点,这就表明了没有路径能多于任何其他路径的两倍长。
3、算法分析 首先,当插入第一个节点时,这个节点就是根节点,所以必须是黑色的。
在增加新的节点的时候,我们先默认新节点都是红色的。为什么呢?因为插入一个红色节点违背红黑规则的可能性比插入一个黑色节点的要小。插入一个红色节点,肯定不会改变树的黑色高度;另外,如果插入节点的父节点是黑色节点,不会违背父子节点同时为红色的规则,如果插入节点的父节点是红色节点才会违背这一规则,这个时候就需要对树进行变换来适应规则。还有一点就是,违背规则3(父子节点都是红色)比违背规则4(黑色高度不同)要容易修正。
下面我们来进行一些实验,依次插入50,25,75,就得到了下面这可棵二叉树:
文章图片
2.png 这棵树符合上面所列出的红黑规则,它是一颗平衡树
3.1、颜色变换 接下来,我们再增加一个值为12的新节点。按照二叉搜索树的规则,12应该插入25的左子节点,但是,由于我们默认新节点都是红色的,而25也是红色的,父子节点不能同时都为红色,这时就需要对25的父节点和其父节点的右子节点进行颜色变换,为什么还要对其父节点的右子节点也进行变换呢?试想,如果我们只将25变换成黑色而75保持红色不变,那么插入新节点之后,根节点的左子树的黑色高度势必会比右子树大1,这就违背了红黑规则 。
实际上,这种情况下,一般也需要将25的父节点变换颜色,因为此时25的父节点是根节点,所以保持黑色不变。
插入12之后,生成下面的红黑树:
文章图片
3.png 这时二叉树显得有一点不平衡,但是仍然符合红黑规则。
接下来,我们再插入一个值为6的新节点。
文章图片
4.png 问题出现了,6的父节点为12,红色,而新节点也是红色,违背了红黑原则。如果将6变换成黑色,则6到根的黑色高度为3,而75到根的高度为2,同样违背了红黑规则。
按照上面的经验,如果我们将12和25的颜色都变换掉呢?
文章图片
5.png 可见,仍然违背了红黑规则。
就是说,一棵树如果超过了两层不平衡(一边的子树比另一边的子树高两层以上),是不可能满足红黑规则的,因为如果一条路径上的节点数比另一条路径上的节点数多一个以上,那它或者有更多的黑色节点,或者有两个相邻的红色节点,都会违背红黑规则。
我们陷入了一个困境,看来仅靠颜色变换是无法走出这个死胡同的。这时候就需要通过另外的途径来对红黑树进行变换了。
3.2 旋转 为了平衡一棵树,需要重新手动地排列节点,如果大部分节点都在根的左侧,就需要把一些节点移到右侧,成为左旋;如果大部分节点都在根的右侧,就需要把一些节点移到左侧,成为右旋。这里所说的选左旋与右旋是相对于一棵子树的根节点而言的。
来看一个实例:
文章图片
6.png 以根为中心,进行一次右旋之后的结果如下:
文章图片
7.png 右旋的时候,以50为中心,周围的节点进行了顺时针旋转。
如果我们已上图中的25为中心,进行一次逆向操作——左旋,将25周围的节点进行逆时针旋转,就又回到了旋转之前的样子。
需要注意的时,旋转的时候仍然要遵循搜索二叉树的规则。比如在对第一幅图中的树进行右旋的时候,节点25以50为中心顺时针旋转,只能旋转到50的左上方,因为25比50小;而75比50大,不能再沿着顺时针方向转动了,所以它仍然在50的右下方保持相对位置不变。
无论是左旋还是右旋,比该节点小的值只能在该节点则左下方或者左上方;比该节点大的值只能在该节点的右下方或者右上方。这样才符合搜索二叉树的规则。
其实,旋转远比上面的简单例子要复杂,再来看一个例子:
文章图片
8.png 我们对一个两层、五个节点的树进行右旋,75比50大,仍然在50的右下方,而25比50小,从50的左下方移到了50的左上方,12作为25的左子节点没有变,但是所在层数有第三层变成了第二层。可以发现,以50为中心右旋之后,50右侧的节点都下降了一个层级(包括50本身),而50左侧的节点都上升了一个层级。
但是,其中发生了一件奇怪的事,就是37从直观上来看,放生了横向移动,从左子树移动到了右子树。让我们分析一下出现这种情况的原因:左子树集体升高了一个层级,37作为25的右子节点,本来也应该水涨船高,随着50上升一个层级,但是,此时25的右子节点已经被50先入为主的占据了,如此一来,37只好沿着右子树重新找家,最后就只能委屈地在50的左下方扎根。
在上例中中,37就被称为50节点右旋操作的内侧子孙,而12就是外侧子孙。
其实,整棵子树也会发生集体的移动,比如:
文章图片
9.png 注意,这种横向移动只会发生在内侧子孙身上。
4、插入 插入或删除操作,都有可能改变红黑树的平衡性,利用颜色变化与旋转这两大法宝就可应对所有情况,将不平衡的红黑树变为平衡的红黑树。
在进行颜色变化或旋转的时候,往往要涉及祖孙三代节点:
- X:表示操作的基准节点
- P:代表X的父节点
- G:代表X的父节点的父节点
- 沿着树查找插入点,如果查找过程中发现某个黑色节点的两个子节点都是红色,则执行一次颜色变换(父节点变为红色,而两个红色子节点变为黑色)。
- 第1步中,不会改变子树的黑色高度,但是可能会出现颜色冲突(红-红颜色冲突),执行一次或两次旋转即可解决。设红色子节点为X,红色父节点为P,旋转次数由X是G的内侧子孙还是外侧子孙决定。
- 找到插入点之后,设X为新插入的节点。如果P是黑色的,则不需要做任何改变,插入完成。
- 如果P是红色的,则发生了红-红颜色冲突,需要做两次颜色变化,如果X为G的外侧子孙,再进行一次旋转;如果X为G的内侧子孙,再进行两次旋转。最终都可使树变为平衡的红黑树。
第1步与第2步看似与插入新节点没关系,其实为了给新节点的插入扫清道路,到后面插入新节点时就会体现出来。
先来看第1步的详细过程:
文章图片
10.png 上图中,查找到P点,发现它的两个子节点都是红色,则进行颜色变换(如果P是根,则保持黑色不变)。这种变换并不会改变从根节点经P到叶节点或者空节点的路径上的黑色节点总数,即不会改变其黑色高度。将P、X1、X2看做三角形的三个顶点,颜色变换之前,经过此三角形时会增加一个黑色节点,颜色变换之后,P变成了红色,X1、X2变成了黑色,不论是经过X1还是经过X2,还是会增加一个黑色节点。
如果P的父节点是黑色,则不会出现任何问题,但是,如果P的父节点也是红色,就会发生红-红颜色冲突,需要通过旋转来修正。发生颜色冲突时有两种情况需要区别对待。
注意,这时候我们选定红-红颜色冲突父子节点中的子节点作为基准节点,即X。如果X在P的一侧与P在G的一侧相同,X即为G的外侧子孙,反之,则为内侧子孙。
4.1 情况1:X为外侧子孙节点。
文章图片
11.png 上图中,表示的是颜色变换之后的情况,12跟25节点发生了颜色冲突,12为50的外侧子孙。
在这种情况需要采取三步操作:
- 改变G的颜色
- 改变P的颜色
- 以G为中心进行向X上升的方向旋转(本例中是右旋)
文章图片
12.png 奇迹发生了,树突然之间平衡了,而且是符合红黑规则的。
需要注意的是,在本例中,由于25是50的左子节点,进行的是右旋操作,加入它是右子节点,则需要进行左旋操作。无论是左旋还是右旋,都是向着X上升的方向旋转。
4.2 情况2:X为内侧子孙节点。 修正这种情况比较复杂一点,如果我们采取跟内侧子孙一样的做法,X不会上移而是发生横向移动,使树变得更加不平衡。因此需要一种不同的方法来解决。
我们先要用一次旋转让X成为外侧子孙,然后再用一次旋转使树平衡。
这种情况需要进行四步操作:
- 改变G的颜色
- 改变X的颜色
- 以P为中心向X上升的方向旋转
- 以G为中心向X上升的方向旋转
文章图片
13.png 至此,前期工作已经完成,下面进行新节点的插入。在插入环节,我们以新节点为基准点,即X。
在前面已经说过,我们总是默认新节点为红色。那么,找到插入点的时候,会有两种情况,一种是X的父节点为P为黑色,直接插入即可(因为插入一个红色新节点既不会影响树的黑色高度,也不会发生颜色冲突);另一种情况是X的父节点P也为红色,插入后会发生红-红颜色冲突,需要通过颜色变换与旋转来修正。
发生颜色冲突的时候,根据X是内侧子孙还是外侧子孙分别对待,处理方法与上面提到的方法类似。
外侧子孙:
文章图片
14.png 内侧子孙:
文章图片
15.png 下面我们来讨论一下,是否还有其他情况。
假如X有一个兄弟节点S,即P的另一个子节点,会使任何需要的旋转更加复杂。如果P为黑色,无论X有没有兄弟节点,都不需要旋转;如果P为红色,则插入之前,P不可能有一个单独的黑色子节点,因为这样会使S和空子节点的黑色高度不一样。综上,插入新节点之后,不会出现X存在兄弟节点而且需要旋转修正的情况。
假如P有一个兄弟节点,即X的叔节点U,也会使任何需要的旋转更加复杂。如果P为黑色,X插入后不要要做任何旋转;如果P为红色,则U必须为红色,否则,G到P的黑色高度与G到U的黑色高度就不同了。但是,有两个红色子节点的父节点在插入之前我们已经处理掉了,所以这种情况也不会存在。综上,插入新节点之后,不会出现P存在兄弟节点且需要旋转修正的情况。
到现在,就明白为什么要在寻找插入点的过程中,把有两个红色子节点的父节点的颜色变换掉,一方面是为了使树更加平衡,另一方面是大大简化了插入后的旋转操作。
5、删除 删除一个节点同样有可能改变树的平衡性,而且,删除所造成的不平衡性比插入所造成的平衡性的修正更加复杂。
化繁为简是算法分析中一个常用的方法。下面我们就根据欲删除节点的所有情况逐项讨论,而欲删除节点有两种可能的颜色,也需要分别对待。
为简化讨论,我们以欲删除节点在左侧的情况为例进行修正,如果欲删除节点在右侧,进行镜像地修正操作即可。
5.1 欲删除节点是叶子节点,即左子节点、右子节点都为空。 5.1.1 欲删除节点为红色,父节点必为黑色,必无兄弟节点。
只有下图所示两种情况,带黄色边框的为欲删除的节点:
文章图片
16.png 将该节点直接删除即可,所在子树的黑色高度不变,不会影响红黑树的性质。
5.1.2 欲删除节点为黑色,父节点可红可黑,必存在兄弟节点。
为什么说黑色节点必存在兄弟节点呢?如果一个黑色节点不存在兄弟节点,无论父节点是红是黑,则从该节点到父节点会比空子节点到父节点少一个黑色高度,所以这种情况是不存在的。
下面就可能存在的情况逐个分析:
5.1.2.1 父节点是红色,则兄弟节点必为黑色 5.1.2.1.1兄弟节点没有左子树,修正策略如下:
文章图片
17.png 上图中,虚线代表其连接的子节点可在可不在,对修正过程无影响。
5.1.2.1.2兄弟节点有左子树,修正策略如下:
文章图片
18.png 5.1.2.2 父节点是黑色,则兄弟节点可红可黑 5.1.2.2.1 兄弟节点为红色,则它必存在两个黑色子节点。 与左侄节点是否有子节点相关,三种情况。
(1)如果左侄节点有右子节点,修正策略如下:
文章图片
19.png (2)如果左侄节点没有右子节点,只有左子节点,修正策略如下:
文章图片
20.png 这时,左侄节点就有了右子节点,再进行与上一种情况一样的修正。
(3)如果左侄节点为叶子节点,修正策略如下:
文章图片
21.png 5.1.2.2.2 兄弟节点为黑色 与兄弟节点是否有子节点相关。
如果兄弟节点有右子节点,修正策略如下:
文章图片
22.png 如果兄弟节点无右子节点,只有左子节点,修正策略如下:
文章图片
23.png 如果兄弟节点既无左子节点,也无右子节点,如下图所示:
文章图片
24.png 此时,依靠该子树的自身是无法解决的,因为该子树的黑色高度为2,如果将带黄色边框的节点删除,无论如何变换颜色、旋转都无法使该子树恢复黑色高度。
有两种解决思路:
第一种是借助欲删除节点的祖父节点及祖父节点的子树来修正,只要祖父节点与祖父节点的另一颗子树中含有红色节点,就能通过颜色变化和旋转来使。但是,如果祖父节点与祖父节点的另一颗子树中的节点全为黑色:
文章图片
25.png 这时,依靠该子树本身也是无法解决的,还需要借助更上层的节点。层层传递,直到根节点。如果到根节点还是不能解决,就需要采用第二种思路:降低黑色高度。
文章图片
26.png 但就这颗子树来说,确实是符合红黑规则的,但是,子树的黑色高度降低,会影响到整棵红黑树的黑色高度。
文章图片
27.png 可以看到,变换之后,左子树的黑色高度为1,而右子树的黑色高度是2,违背了红黑规则。这时候就需要同时降低右子树的黑色高度,并层层向上传递,直到根节点,最终使整棵树的黑色高度降低。
但是,有的右子树是无法降低黑色高度的,比如:
文章图片
28.png 上图中的右子树就无法降低黑色高度,但是这种情况下,右子树必存在红色节点,可以通过第一种思路解决。
总之,结合两种思路,总是是可以解决“黑三角”这种顽固问题的。
5.2 欲删除节点只有一个子节点,该节点必为黑色,子节点必为红色 红色节点要么没有子节点,要么有两个黑色节点,所以该节点不可能为红色。
如果该节点只有一个黑色节点,则黑色的子节点到该节点的黑色高度与空子节点到该点的黑色高度不一致,所以子节点只能为红色。
综上,只有下图所示的两种情况会出现。
文章图片
29.png 只需要将红色子节点涂黑,上移到被删除的节点位置即可。
文章图片
30.png 5.3 欲删除节点既有左子节点,又有右子节点。 红黑树也属于二叉搜索树,所以要先找到欲删除节点的后继节点。后继节点的寻找过程如下:从该点的右子节点开始,如果有左子节点则跳到左子节点,层层向下,直到某个子节点没有左子节点为止。实际上就是找到比欲删除节点的关键字值大的集合中的最小值。这部分内容在二叉搜索树中已经介绍过,不再赘述。
第一步,我们要将欲删除节点与后继节点中的数据对换,如此一来,节点的删除操作就转移到了后继节点上来了。
文章图片
31.png 上图中,经过数据交换之后,没有改变任何节点的颜色,现在要删除的是带黄色边框的红色节点。前面已经论证,后继节点肯定是一个没有左子节点的节点,即是一个叶子节点或者只有一个右子节点的节点,而这两种情况的删除我们在之前已经分析过并给出了解决方法。
至此,红黑树所有可能出现的删除情况都已经讨论完毕。
推荐阅读
- JAVA(抽象类与接口的区别&重载与重写&内存泄漏)
- Docker应用:容器间通信与Mariadb数据库主从复制
- 《真与假的困惑》???|《真与假的困惑》??? ——致良知是一种伟大的力量
- 第326天
- Shell-Bash变量与运算符
- 画解算法(1.|画解算法:1. 两数之和)
- 逻辑回归的理解与python示例
- Guava|Guava RateLimiter与限流算法
- 我和你之前距离
- CGI,FastCGI,PHP-CGI与PHP-FPM