TreeMap 插入 key-value 同样是用 put 方法
put
1 | public V put(K key, V value) { |
fixAfterInsertion
这里我们把这个方法的代码拆分来看, 至于红黑树的调整原理可以看 HashMap-详解五, 这里不在赘述.
插入的新节点要是红色, 所以要先修改节点颜色
1
x.color = RED;
只有红父才需要调整, 所以节点不会是根节点, 循环结构如下:
1
2
3while (x != null && x != root && x.parent.color == RED) {
...
}对于新节点是位于祖父节点的左子树, 叔叔节点是红色, 那么只要改变颜色即可, 同时当前节点指向祖父节点, 循环向上遍历
1
2
3
4
5
6
7
8
9if (parentOf(x) == leftOf(parentOf(parentOf(x)))) {
Entry<K,V> y = rightOf(parentOf(parentOf(x)));
if (colorOf(y) == RED) {
setColor(parentOf(x), BLACK);
setColor(y, BLACK);
setColor(parentOf(parentOf(x)), RED);
x = parentOf(parentOf(x));
}
}对于新节点是位于祖父节点的左子树, 叔叔节点是黑色, 如果新节点位于父节点的右子树, 先左旋, 然后统一右旋, 当然也要改变节点颜色
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17if (parentOf(x) == leftOf(parentOf(parentOf(x)))) {
Entry<K,V> y = rightOf(parentOf(parentOf(x)));
if (colorOf(y) == RED) {
...
} else {
// 看这里
// 左旋之后, 当前节点是原来的父节点
// 这样就可以相当于插入一个新的左节点, 跟下面的右旋完美结合
if (x == rightOf(parentOf(x))) {
x = parentOf(x);
rotateLeft(x);
}
setColor(parentOf(x), BLACK);
setColor(parentOf(parentOf(x)), RED);
rotateRight(parentOf(parentOf(x)));
}
}对于新节点是位于祖父节点的右子树, 叔叔节点是红色, 直接修改颜色即可
1
2
3
4
5
6
7Entry<K,V> y = leftOf(parentOf(parentOf(x)));
if (colorOf(y) == RED) {
setColor(parentOf(x), BLACK);
setColor(y, BLACK);
setColor(parentOf(parentOf(x)), RED);
x = parentOf(parentOf(x));
}对于新节点是位于祖父节点的右子树, 叔叔节点是黑色, 如果新节点位于父节点的左子树, 先右旋, 然后统一左旋, 同样要修改颜色
1
2
3
4
5
6
7
8
9
10
11
12Entry<K,V> y = leftOf(parentOf(parentOf(x)));
if (colorOf(y) == RED) {
...
} else {
if (x == leftOf(parentOf(x))) {
x = parentOf(x);
rotateRight(x);
}
setColor(parentOf(x), BLACK);
setColor(parentOf(parentOf(x)), RED);
rotateLeft(parentOf(parentOf(x)));
}
完整代码如下:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40private void fixAfterInsertion(Entry<K,V> x) {
x.color = RED;
while (x != null && x != root && x.parent.color == RED) {
if (parentOf(x) == leftOf(parentOf(parentOf(x)))) {
Entry<K,V> y = rightOf(parentOf(parentOf(x)));
if (colorOf(y) == RED) {
setColor(parentOf(x), BLACK);
setColor(y, BLACK);
setColor(parentOf(parentOf(x)), RED);
x = parentOf(parentOf(x));
} else {
if (x == rightOf(parentOf(x))) {
x = parentOf(x);
rotateLeft(x);
}
setColor(parentOf(x), BLACK);
setColor(parentOf(parentOf(x)), RED);
rotateRight(parentOf(parentOf(x)));
}
} else {
Entry<K,V> y = leftOf(parentOf(parentOf(x)));
if (colorOf(y) == RED) {
setColor(parentOf(x), BLACK);
setColor(y, BLACK);
setColor(parentOf(parentOf(x)), RED);
x = parentOf(parentOf(x));
} else {
if (x == leftOf(parentOf(x))) {
x = parentOf(x);
rotateRight(x);
}
setColor(parentOf(x), BLACK);
setColor(parentOf(parentOf(x)), RED);
rotateLeft(parentOf(parentOf(x)));
}
}
}
root.color = BLACK;
}
左旋和右旋实际是修改节点的左右子树和父节点, 可以通过自己画图来更好理解节点的交换过程
rotateLeft
1 | // 左旋 |
rotateRight
1 | // 右旋 |
如果有疑问欢迎来 Issues 探讨