对于红黑树的删除,看了数据结构的书,也看了很多网上的讲解和实现,但都不满意。很多讲解都是囫囵吞枣,知其然,不知其所以然,讲的晦涩难懂。
红黑树是平衡二叉树的一种,其删除算法是比较复杂的,因为删除后还要保持红黑树的特性。红黑树的特性如下:
- 节点是红色或黑色。
- 根是黑色。
- 所有叶子都是黑色(叶子是NIL节点)。
- 每个红色节点必须有两个黑色的子节点。(从每个叶子到根的所有路径上不能有两个连续的红色节点。)
- 从任一节点到其每个叶子的所有简单路径都包含相同数目的黑色节点(简称黑高)。
因此,从红黑树最基础的特性出发,抛开教科书和网上的算法,画了无数张图,分析了多种可能的情况以后,经过归纳提炼,实现了不同于教科书上的删除算法。
经过多次画图证明以后,笔者发现,红黑树的删除算法不是唯一的,不管如何调整,只要保证删除后还是一颗红黑树即可。
因此,笔者实现的 删除思路和算法如下:
1. 删除转移:(这部分是大路货,不是自己实现的)
- 如果被删除节点有两个非空子节点,则用后继节点的值代替该节点的值,这样演变成了删除后继节点;否则转下一条;
- 如果被删除节点一个或两个孩子都为空:若有非空孩子,则用非空孩子节点替代之;若无,直接删除;
- 删除后继节点:后继节点的左孩子节点一定为空,右孩子可能为空;处理如上一条;
删除转移的目的是为了简化删除操作,更是为了简化修复操作。因为删除转移后,最终待删除的节点最多只会有一个非空孩子。
2. 删除后修复:
2.1 简单的情况:
- 若被删除节点为红色节点,不需修复;此时该节点一定为红色的叶子节点(可根据红黑树的特性证明);
- 若被删除的节点为黑色节点,且有一个非空子节点,则将其非空子节点颜色涂黑即可;
对于以上两种简单的情况,做个说明:根据红黑树特性,非空子节点一定为红色节点,否则将违反特性;根据红黑树特性,在删除前,一颗红黑树不可能出现以下几种情况:
(图片来自网络,感谢原作者。)
2.2 复杂的情况:删除后需要修复的。
只有当被删除的节点为黑色叶子节点时,导致该节点所在的分支,少了一个黑色节点,树不再平衡,因此需要修复。
修复的整体思路是:
- 如果该节点的父节点、或兄弟节点、或兄弟节点的特定方向的子节点 中,有红色节点,则将此红色节点旋转过来,通过旋转、涂黑操作,保持自父节点以来的树的平衡;
- 如果不满足上述条件,则通过旋转和变色操作,使其兄弟分支上也减少一个黑色节点,这样自父节点以来的分支保持了平衡,满足了条件,但对于父节点来说,其整个分支减少了一个黑色节点,需要递归向上处理,直至重新平衡,或者到达根节点;
掌握了整体思路以后,就可编码实现了,编码中用了一些小技巧,合并了一些情况,代码比较简单易懂,阅读者可以根据代码的情况自己画图证明:
说明:代码为dart语言实现,dart语法基本与Java一致,不清楚的地方可以参考:
https://www.dartlang.org/guides/language/language-tour
// 删除
bool delete(E value) {
var node = find(value);
if (node == null) return false;
_delete(node);
_nodeNumbers--;
return true;
} // 删除转移 并修复
void _delete(RBTNode<E> d) {
if (d.left != null && d.right != null) {
var s = _successor(d);
d.value = s.value;
d = s;
} var rp = d.left ?? d.right;
rp?.parent = d.parent;
if (d.parent == null)
_root = rp;
else if (d == d.parent.left)
d.parent.left = rp;
else
d.parent.right = rp; if (rp != null)
rp.paintBlack();
else if (d.isBlack && d.parent != null)
_fixAfterDelete(d.parent, d.parent.left == null);
} RBTNode<E> _successor(RBTNode<E> d) =>
d.right != null ? _minNode(d.right) : d.left; RBTNode<E> _minNode(RBTNode<E> r) => r.left == null ? r : _minNode(r.left); // fix up after delete
void _fixAfterDelete(RBTNode<E> p, bool isLeft) {
var ch = isLeft ? p.right : p.left;
if (isLeft) { // 如果被删除节点是父节点p的左分支;
if (p.isRed) { // 如果父节点为红,则兄弟节点ch一定为黑;
if (ch.left != null && ch.left.isRed) {
p.paintBlack();
_rotateRight(ch);
}
_rotateLeft(p);
} else if (ch.isRed) { // 兄弟节点为红,此时兄弟节点一定有两个非空黑色子节点;
p.paintRed();
ch.paintBlack();
_rotateLeft(p);
_fixAfterDelete(p, true); // 变换为父节点为红的情况,递归处理;
} else if (ch.left != null && ch.left.isRed) { // 父、兄均为黑,兄有红色左孩子;
ch.left.paintBlack();
_rotateRight(ch);
_rotateLeft(p);
} else { // 父兄均为黑,将父分支左右均减少一个黑节点,然后递归向上处理;
p.paintRed();
_rotateLeft(p);
if (ch.parent != null) _fixAfterDelete(ch.parent, ch == ch.parent.left);
}
} else { // symmetric
if (p.isRed) {
if (ch.right != null && ch.right.isRed) {
p.paintBlack();
_rotateLeft(ch);
}
_rotateRight(p);
} else if (ch.isRed) {
p.paintRed();
ch.paintBlack();
_rotateRight(p);
_fixAfterDelete(p, false);
} else if (ch.right != null && ch.right.isRed) {
ch.right.paintBlack();
_rotateLeft(ch);
_rotateRight(p);
} else {
p.paintRed();
_rotateRight(p);
if (ch.parent != null) _fixAfterDelete(ch.parent, ch == ch.parent.left);
}
}
}
旋转操作的代码:
void _rotateLeft(RBTNode<E> node) {
var r = node.right, p = node.parent;
r.parent = p;
if (p == null)
_root = r;
else if (p.left == node)
p.left = r;
else
p.right = r; node.right = r.left;
r.left?.parent = node;
r.left = node;
node.parent = r;
} void _rotateRight(RBTNode<E> node) {
var l = node.left, p = node.parent;
l.parent = p;
if (p == null)
_root = l;
else if (p.left == node)
p.left = l;
else
p.right = l; node.left = l.right;
l.right?.parent = node;
l.right = node;
node.parent = l;
}