红黑树除了符合二叉搜索树的基本规则外,还添加了如下特性:算法
红黑树的相对平衡数据结构
前面5条规则的约束确保了如下红黑树的关键特性:ide
为何能够作到最长路径不超过最短路径的两倍呢?递归
插入一个新节点时,有可能树再也不平衡,能够经过三种方式的变换使树保持平衡:ip
为了从新符合红黑树的规则,须要把红色节点变为黑色,或者把黑色节点变为红色;v8
插入的新节点一般都是红色节点:get
当插入的节点为红色的时候,大多数状况不违反红黑树的任何规则;效率
红色节点虽然可能致使红红相连的状况,可是这种状况能够经过颜色调换和旋转来调整;原理
以节点X为根逆时针旋转二叉搜索树,使得父节点原来的位置被本身的右子节点替代,左子节点的位置被父节点替代;数据结构与算法
详解:
如上图所示,左旋转以后:
除此以外,二叉搜索树左旋转以后仍为二叉搜索树:
以节点X为根顺时针旋转二叉搜索树,使得父节点原来的位置被本身的左子节点替代,右子节点的位置被父节点替代;
详解:
如上图所示,右旋转以后:
除此以外,二叉搜索树右旋转以后仍为二叉搜索树:
首先须要明确,在保证知足红黑树5条规则的状况下,新插入的节点必然是红色节点。
为了方便说明,规定如下四个节点:新插入节点为N(Node),N的父节点为P(Parent),P的兄弟节点为U(Uncle),U的父节点为G(Grandpa),以下图所示:
当插入的新节点N位于树的根上时,没有父节点。
这种状况下,只须要将红色节点变为黑色节点便可知足规则2 。
新界点N的父节点P为黑色节点,此时不须要任何变化。
此时既知足规则4也知足规则5。尽管新节点是红色的,可是新节点N有两个黑色节点NIL,因此通向它的路径上黑色节点的个数依然相等,所以知足规则5 。
节点P为红色,节点U也为红色,此时节点G必为黑色,即父红叔红祖黑。
在这种状况下须要:
即变为父黑叔黑祖红,以下图所示:
可能出现的问题:
节点P是红色节点,节点U是黑色节点,而且节点N为节点P的左子节点,此时节点G必定是黑色节点,即父红叔黑祖黑。
在这种状况下须要:
节点P是红色节点,节点U是黑色节点,而且节点N为节点P的右子节点,此时节点G必定是黑色节点,即父红叔黑祖黑。
在这种状况下须要:
接着能够按照状况4进行处理:
先变色:将N1节点的父节点P1变为黑色,将祖父节点G变为红色;
最后将节点N1和P1变换回来,完成节点N的插入,如图 f 所示;
在二叉树中依次插入节点:10,9,8,7,6,5,4,3,2,1 。
若是直接采用普通的二叉搜索树,节点所有插入后是这样的:
是一个严重的不平衡树,至关于一个链表,不能体现出二叉搜索树的高效率。而按照红黑树的五条规则插入节点就能最大程度保证搜索二叉树是一棵平衡树。如下为过程详解:为了方便解释省略了部分成黑树的叶子节点(NIL)
符合状况1:
符合状况2:
快速判断属于状况3仍是状况4的方法:
重新插入的节点N出发,按图示箭头通过的四个节点,若为红红黑红3个红色节点则为状况3,若为红红黑黑两个红色节点则为状况4;
符合状况4:
符合状况3:
符合状况4:
符合状况3:
符合状况4:
第一次变换:符合状况3:
变换以后发现5和7为相连的两个红色节点,因而把以5为根的整个子树当作一个新插入的节点N1,再进行第二次变换。
第二次变换:符合状况4:
最后复原N1获得变换后的红黑树:
符合状况4:
第一次变换:符合状况3:
变换以后发现3和5为相连的两个红色节点,因而把以3为根的整个子树当作一个新插入的节点N1,再进行第二次变换。
第二次变换:符合状况3:
变换以后发现根节点7为红色不符合规则2,因此把以7为根节点的红黑树当作一个新插入的节点N2,再进行第三次变换。
第三次变换:符合状况1:
由此,完成了1~10节点的插入,虽然没有遇到状况5,不过状况5通过左旋转的操做即可转换为状况4,原理同样。以下图所示,将这棵红黑树的叶子节点NIL补全以后,经检验知足红黑树的五条规则,而且基本属于平衡树,效率较高。
红黑树的删除操做结合了复杂的二叉树的删除操做和复杂的红黑树的插入规则,总体来讲难度很是大,篇幅较长,这里暂不进行探讨。
参考资料:JavaScript数据结构与算法