服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - Java教程 - 通过java.util.TreeMap源码加强红黑树的理解

通过java.util.TreeMap源码加强红黑树的理解

2021-02-07 12:09fireway Java教程

通过分析java.util.TreeMap源码来对经典问题红黑树加强理解和理清思路。

在此之前,服务器之家已经为大家整理了很多关于经典问题红黑树的思路和解决办法。本篇文章,是通过分析java.util.treemap源码,让大家通过实例来对红黑树这个问题有更加深入的理解。

本篇将结合jdk1.6的treemap源码,来一起探索红-黑树的奥秘。红黑树是解决二叉搜索树的非平衡问题。

当插入(或者删除)一个新节点时,为了使树保持平衡,必须遵循一定的规则,这个规则就是红-黑规则: 
1) 每个节点不是红色的就是黑色的 
2) 根总是黑色的 
3) 如果节点是红色的,则它的子节点必须是黑色的(反之倒不一定必须为真) 
4) 从跟到叶节点或者空子节点的每条路径,必须包含相同数目的黑色节点

插入一个新节点

红-黑树的插入过程和普通的二叉搜索树基本一致:从跟朝插入点位置走,在每个节点处通过比较节点的关键字相对大小来决定向左走还是向右走。

?
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
public v put(k key, v value) {
entry<k,v> t = root;
int cmp;
entry<k,v> parent;
comparable<? super k> k = (comparable<? super k>) key;
do {
parent = t;
cmp = k.compareto(t.key);
if (cmp < 0) {
t = t.left;
} else if (cmp > 0) {
t = t.right;
} else {
// 注意,return退出方法
return t.setvalue(value);
}
} while (t != null);
entry<k,v> e = new entry<k,v>(key, value, parent);
if (cmp < 0) {
parent.left = e;
} else {
parent.right = e;
}
fixafterinsertion(e);
size++;
modcount++;
return null;
}

但是,在红-黑树种,找到插入点更复杂,因为有颜色变换和旋转。fixafterinsertion()方法就是处理颜色变换和旋转,需重点掌握它是如何保持树的平衡(use rotations and the color rules to maintain the tree's balance)。

下面的讨论中,使用x、p、g表示关联的节点。x表示一个特殊的节点, p是x的父,g是p的父。

x is a node that has caused a rule violation. (sometimes x refers to a newly inserted node, and sometimes to the child node when a parent and child have a redred conflict.)

on the way down the tree to find the insertion point, you perform a color flip whenever you find a black node with two red children (a violation of rule 2). sometimes the flip causes a red-red conflict (a violation of rule 3). call the red child x and the red parent p. the conflict can be fixed with a single rotation or a double rotation, depending on whether x is an outside or inside grandchild of g. following color flips and rotations, you continue down to the insertion point and insert the new node.

after you've inserted the new node x, if p is black, you simply attach the new red node. if p is red, there are two possibilities: x can be an outside or inside grandchild of g. if x is an outside grandchild, you perform one rotation, and if it's an inside grandchild, you perform two. this restores the tree to a balanced state.

按照上面的解释,讨论可分为3个部分,按复杂程度排列,分别是: 
1) 在下行路途中的颜色变换(color flips on the way down) 
2) 插入节点之后的旋转(rotations after the node is inserted) 
3) 在向下路途上的旋转(rotations on the way down)

在下行路途中的颜色变换(color flips on the way down)

here's the rule: every time the insertion routine encounters a black node that has two red children, it must change the children to black and the parent to red (unless the parent is the root, which always remains black)

the flip leaves unchanged the number of black nodes on the path from the root on down through p to the leaf or null nodes.

尽管颜色变换不会违背规则4,但是可能会违背规则3。如果p的父是黑色的,则p由黑色变成红色时不会有任何问题,但是,如果p的父是红色的,那么在p的颜色变化之后,就有两个红色节点相连接了。这个问题需要在继续向下沿着路径插入新节点之前解决,可以通过旋转修正这个问题,下文将会看到。

插入节点之后的旋转(rotations after the node is inserted)

新节点在插入之前,树是符合红-黑规则,在插入新节点之后,树就不平衡了,此时需要通过旋转来调整树的平衡,使之重新符合红-黑规则。

可能性1:p是黑色的,就什么事情也不用做。插入即可。

可能性2:p是红色,x是g的一个外侧子孙节点,则需要一次旋转和一些颜色的变化。 
以插入50,25,75,12,6为例,注意节点6是一个外侧子孙节点,它和它的父节点都是红色。 

通过java.util.TreeMap源码加强红黑树的理解

在这个例子中,x是一个外侧子孙节点而且是左子节点,x是外侧子孙节点且为右子节点,是一种与此对称的情况。通过用50,25,75,87,93创建树,同理再画一画图,这里就省略了。

可能性3:p是红色,x是g的一个内侧子孙节点,则需要两次旋转和一些颜色的改变。 
以插入50,25,75,12,18为例,注意节点18是一个内侧子孙节点,它和它的父节点都是红色。 

通过java.util.TreeMap源码加强红黑树的理解

在向下路途上的旋转(rotations on the way down)

在插入新节点之前,实际上树已经违背了红-黑规则,所以需要插入新节点之前做调整。所以我们本次讨论的主题是“在向下路途准备插入新节点时,上面先进行调整,使上面成为标准的红黑树后,再进行新节点插入”。

外侧子孙节点

以插入50,25,75,12,37,6,18,3为例,例子中违背规则的节点是一个外侧子孙节点。 

通过java.util.TreeMap源码加强红黑树的理解

内侧子孙节点

以插入50,25,75,12,37,31,43为例,例子中违背规则的节点是一个内侧子孙节点。
通过java.util.TreeMap源码加强红黑树的理解

红-黑树的效率

和一般的二叉搜索树类似,红-黑树的查找、插入和删除的时间复杂度为o(log2n)。

红-黑树的查找时间和普通的二叉搜索树的查找时间应该几乎完全一样。因为在查找过程中并没用到红-黑特征。额外的开销只是每个节点的存储空间都稍微增加了一点,来存储红黑颜色(一个boolean变量)。

?
1
final entry<k, v> getentry(object key) {<br>comparable <? super k > k = (comparable <? super k > ) key;<br>entry<k, v> p = root;<br>while (p != null) {<br>int cmp = k.compareto(p.key);<br>if (cmp < 0) {<br>p = p.left;<br>} else if (cmp > 0) {<br>p = p.right;<br>} else {<br>return p;<br>}<br>}<br>return null;<br>}

插入和删除的时间要增加一个常数因子,因为不得不在下行的路径上和插入点执行颜色变换和旋转。平均起来一次插入大约需要一次旋转。

因为在大多数应用中,查找的次数比插入和删除的次数多,所以应用红-黑树取代普通的二叉搜索树总体上不会增加太多的时间开销。

原文链接:https://www.cnblogs.com/fireway/p/7862577.html

延伸 · 阅读

精彩推荐