红黑树其实并不难,只是你还没看过ta

红黑树概念

R-B Tree,全称是Red-Black Tree,又称为“红黑树”,它一种特殊的二叉查找树。红黑树的每个节点上都有存储位表示节点的颜色,可以是红(Red)或黑(Black)。

红黑树的特性

(1)每个节点或者是黑色,或者是红色。

(2)根节点是黑色。

(3)每个叶子节点(NIL)是黑色。 [注意:这里叶子节点,是指为空(NIL或NULL)的叶子节点!]

(4)如果一个节点是红色的,则它的子节点必须是黑色的。

(5)从一个节点到该节点的子孙节点的所有路径上包含相同数目的黑节点。

红黑树左旋

对x进行左旋,意味着,将“x的右孩子”设为“x的父亲节点”;即,将 x变成了一个左节点(x成了为z的左孩子)!。 因此,左旋中的“左”,意味着“被旋转的节点将变成一个左节点”。


f926b9529ad84220b69cb6234e5c0184


LEFT-ROTATE(T, x)

y ← right[x] // 前提:这里假设x的右孩子为y。下面开始正式操作

right[x] ← left[y] // 将 “y的左孩子” 设为 “x的右孩子”,即 将β设为x的右孩子

p[left[y]] ← x // 将 “x” 设为 “y的左孩子的父亲”,即 将β的父亲设为x

p[y] ← p[x] // 将 “x的父亲” 设为 “y的父亲”

if p[x] = nil[T]

then root[T] ← y // 情况1:如果 “x的父亲” 是空节点,则将y设为根节点

else if x = left[p[x]]

then left[p[x]] ← y // 情况2:如果 x是它父节点的左孩子,则将y设为“x的父节点的左孩子”

else right[p[x]] ← y // 情况3:(x是它父节点的右孩子) 将y设为“x的父节点的右孩子”

left[y] ← x // 将 “x” 设为 “y的左孩子”

p[x] ← y // 将 “x的父节点” 设为 “y”


7fd78abd7ba5410095969c56e8962046


红黑树右旋

对x进行右旋,意味着,将“x的左孩子”设为“x的父亲节点”;即,将 x变成了一个右节点(x成了为y的右孩子)! 因此,右旋中的“右”,意味着“被旋转的节点将变成一个右节点”。


8a2e37709a7349a8972e7e4cb439200e


RIGHT-ROTATE(T, y)

x ← left[y] // 前提:这里假设y的左孩子为x。下面开始正式操作

left[y] ← right[x] // 将 “x的右孩子” 设为 “y的左孩子”,即 将β设为y的左孩子

p[right[x]] ← y // 将 “y” 设为 “x的右孩子的父亲”,即 将β的父亲设为y

p[x] ← p[y] // 将 “y的父亲” 设为 “x的父亲”

if p[y] = nil[T]

then root[T] ← x // 情况1:如果 “y的父亲” 是空节点,则将x设为根节点

else if y = right[p[y]]

then right[p[y]] ← x // 情况2:如果 y是它父节点的右孩子,则将x设为“y的父节点的左孩子”

else left[p[y]] ← x // 情况3:(y是它父节点的左孩子) 将x设为“y的父节点的左孩子”

right[x] ← y // 将 “y” 设为 “x的右孩子”

p[y] ← x // 将 “y的父节点” 设为 “x”

添加操作

第一步: 将红黑树当作一颗二叉查找树,将节点插入。

第二步:将插入的节点着色为"红色"。

根据被插入节点的父节点的情况,可以将"当节点z被着色为红色节点,并插入二叉树"划分为三种情况来处理。

① 情况说明:被插入的节点是根节点。

处理方法:直接把此节点涂为黑色。

② 情况说明:被插入的节点的父节点是黑色。

处理方法:什么也不需要做。节点被插入后,仍然是红黑树。

③ 情况说明:被插入的节点的父节点是红色。这种情况下,被插入节点是一定存在非空祖父节点的;进一步的讲,被插入节点也一定存在叔叔节点(即使叔叔节点为空,我们也视之为存在,空节点本身就是黑色节点)。理解这点之后,我们依据"叔叔节点的情况",将这种情况进一步划分为3种情况(Case)。


1f101316ff25453b9283ef1e61207b2b



第三步: 通过一系列的旋转或着色等操作,使之重新成为一颗红黑树。

删除操作

第一步:将红黑树当作一颗二叉查找树,将节点删除。

这和"删除常规二叉查找树中删除节点的方法是一样的"。分3种情况:

① 被删除节点没有儿子,即为叶节点。那么,直接将该节点删除就OK了。

② 被删除节点只有一个儿子。那么,直接删除该节点,并用该节点的唯一子节点顶替它的位置。

③ 被删除节点有两个儿子。那么,先找出它的后继节点;然后把“它的后继节点的内容”复制给“该节点的内容”;之后,删除“它的后继节点”。

第二步:通过"旋转和重新着色"等一系列来修正该树,使之重新成为一棵红黑树。

因为"第一步"中删除节点之后,可能会违背红黑树的特性。所以需要通过"旋转和重新着色"来修正该树,使之重新成为一棵红黑树。

选择重着色3种情况。

① 情况说明:x是“红+黑”节点。

处理方法:直接把x设为黑色,结束。此时红黑树性质全部恢复。

② 情况说明:x是“黑+黑”节点,且x是根。

处理方法:什么都不做,结束。此时红黑树性质全部恢复。

③ 情况说明:x是“黑+黑”节点,且x不是根。

处理方法:这种情况又可以划分为4种子情况。这4种子情况如下表所示:


00d5a39ff7444403b226d5c1a71aa5a8


红黑色的大体思想就是上面描述的那样,里面还有很多情况要考虑,本文只是简单的讲述思想,大家有兴趣的可以深入研究一下,谢谢大家的支持!


196c801cfae84f5994256b01cf9db93b


猜你喜欢

转载自blog.51cto.com/14794073/2496361