1. 定义

红黑树也是二叉查找树,我们知道,二叉查找树这一数据结构并不难,而红黑树之所以难是难在它是自平衡的二叉查找树,在进行插入和删除等可能会破坏树的平衡的操作时,需要重新自处理达到平衡状态。红黑树是一种含有红黑结点并能自平衡的二叉查找树,又称黑色完美平衡。

动画演示:https://rbtree.phpisfuture.com/

2. 节点称呼

3. 性质

  • 每个节点要么是黑色,要么是红色。

  • 根节点一定是黑色。

  • 每个叶子节点(nil或null)都是黑色的。

  • 每个红节点的两个子节点一定是黑色的。(不可以同时存在两个相连的红结点,即:红节点的父结点与子结点都是黑的)

  • 从任意节点出发到每个叶子节点的路径都包含相同个数的黑色节点。

    * 如果一个结点存在黑子结点,那么该结点肯定有两个子结点。
    
    * 黑色完美平衡。

下面是一棵简单的红黑树,Nil(java中为null)是叶子节点并为黑色:

上图中的红黑树并不是完美平衡的二叉查找树,P节点的左边比右边高,但是左右黑色的层数是相等的,任意一个结点到叶子节点的黑色节点数都相同(性质5),也被成为黑色完美平衡。

4. 红黑树的自平衡

4.1 左旋

以某个结点作为支点(旋转结点),其右子结点变为旋转结点的父结点,右子结点的左子结点变为旋转结点的右子结点,其他结点保持不变。

4.2 右旋

以某个结点作为支点(旋转结点),其左子结点变为旋转结点的父结点,左子结点的右子结点变为旋转结点的左子结点,其他结点保持不变。

4.3 变色

结点的颜色由红变黑或由黑变红。

5. 红黑树的查找

红黑树是一颗二叉平衡树,查找不会破坏平衡性,所以和二叉平衡术查找方式一致。

  • 从根节点开始查找,为空就返回null,为当前值就返回,否则继续向下查找。
  • 如果当前节点的key为要查找的节点的key,那么直接返回当前值。
  • 如果当前节点的key大于要查找的节点的key,那么继续向当前节点的左子节点查找。
  • 如果当前节点的key小于要查找的节点的key,那么继续向当前节点的右子节点查找。

6. 红黑树的插入

插入会破坏红黑树的黑色完美平衡,所以插入第一步要找到要插入的位置进行插入,第二步进行自平衡。

6.1 查找插入位置

所有插入操作都是在叶子结点进行的。

  • 插入节点的颜色肯定为红色。因为插入节点为黑色,就会破坏黑色完美平衡,使得到叶子节点的黑色数+1,而红色不会破坏。
  • 基本与红黑树的查找相同:

从根节点开始,如果根节点为空,则插入在根节点,否则根节点为当前节点。

  • 如果当前节点为null,则返回当前节点的父节点进行插入。
  • 如果当前节点的key等与插入节点的key,则更新当前节点的value。
  • 如果当前节点的key大于插入节点的key,则继续向当前节点的左子节点继续查找。
  • 如果当前节点的key小于插入节点的key,则继续向当前节点的右子节点继续查找。

6.2 插入的自平衡

插入主要指针指向插入结点,通过4. 红黑树的自平衡将红黑树达到的平衡即可

左旋

条件:当前节点的父节点是红色 & 当前节点的叔叔节点是黑色或者不存在 & 当前结点是其父节点的右子结点。

步骤:

  • 将父节点左旋
  • 将指针指向父结点

右旋

条件:当前节点的父节点是红色 & 当前节点的叔叔节点是黑色或者不存在 & 当前结点是其父节点的左子结点。

步骤:

  • 将父节点变为黑色
  • 将祖父结点变为红色
  • 将祖父结点右旋
  • 将指针指向祖父结点

变色

条件:当前节点的父节点是红色并且当前节点的叔叔节点也是红色。

步骤:

  • 当前结点是根结点直接变为黑色
  • 当前结点不是根结点
  • 将父节点与叔叔节点变为黑色
  • 将祖父结点变为红色
  • 将指针指向祖父结点

JDK1.8中插入自平衡的源码实现:

private 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)))) {
// y是插入节点的祖父节点的右子节点(叔叔节点)
Entry<K,V> y = rightOf(parentOf(parentOf(x)));
// y是红色
if (colorOf(y) == RED) {
// 变色处理
setColor(parentOf(x), BLACK);
setColor(y, BLACK);
setColor(parentOf(parentOf(x)), RED);
// 指针指向插入节点的祖父节点
x = parentOf(parentOf(x));
} else {
// y是黑色的
// 插入节点是是父节点的右子节点
if (x == rightOf(parentOf(x))) {
// 父节点左旋
x = parentOf(x);
rotateLeft(x);
}
// 插入节点是是父节点的左节点
setColor(parentOf(x), BLACK);
setColor(parentOf(parentOf(x)), RED);
// 祖父节点右旋
rotateRight(parentOf(parentOf(x)));
}
} else {
// 插入的父节点是右子节点
// y是插入节点的祖父节点的左子节点(叔叔节点)
Entry<K,V> y = leftOf(parentOf(parentOf(x)));
// y是红色
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;
}

7. 红黑树删除

删除操作与插入差不多,查找、删除、自平衡。查找目标结点显然可以复用查找操作,当不存在目标结点时,忽略本次操作;当存在目标结点时,删除后就得做自平衡处理了。删除了结点后我们还需要找结点来替代删除结点的位置,不然子树跟父辈结点断开了,除非删除结点刚好没子结点,那么就不需要替代。

7.1 查找删除位置

基本与红黑树的查找相同:

  • 从根节点开始,如果根节点为空,则删除在根节点,否则根节点为当前节点。
  • 如果当前节点为null,则返回当前节点的父节点进行插入。
  • 如果当前节点的key等与删除节点的key,则找到当前节点。
  • 如果当前节点的key大于删除节点的key,则继续向当前节点的左子节点继续查找。
  • 如果当前节点的key小于删除节点的key,则继续向当前节点的右子节点继续查找。

7.2 删除结点

删除节点的可能情况:

JDK1.8中TreeMap删除可能性源代码实现:

private void deleteEntry(Entry<K,V> p) {
modCount++;
size--; // If strictly internal, copy successor's element to p and then make p
// point to successor.
// 如果删除节点有两个子节点
if (p.left != null && p.right != null) {
// 找到替代节点(很简单,自己看TreeMap源码)
Entry<K,V> s = successor(p);
p.key = s.key;
p.value = s.value;
p = s;
} // p has 2 children // Start fixup at replacement node, if it exists.
// 如果有一个替换节点
Entry<K,V> replacement = (p.left != null ? p.left : p.right); // 如果存在替换节点
if (replacement != null) {
// Link replacement to parent
replacement.parent = p.parent;
if (p.parent == null)
root = replacement;
else if (p == p.parent.left)
p.parent.left = replacement;
else
p.parent.right = replacement; // Null out links so they are OK to use by fixAfterDeletion.
p.left = p.right = p.parent = null; // Fix replacement
if (p.color == BLACK)
fixAfterDeletion(replacement);
}
// 如果删除节点是根节点
else if (p.parent == null) { // return if we are the only node.
root = null;
} else {
// 没有子节点
if (p.color == BLACK)
fixAfterDeletion(p); if (p.parent != null) {
if (p == p.parent.left)
p.parent.left = null;
else if (p == p.parent.right)
p.parent.right = null;
p.parent = null;
}
}
}

7.3 删除后的自平衡

删除自平衡处理:

JDK1.8中TreeMap删除自平衡源代码实现:

private void fixAfterDeletion(Entry<K,V> x) {
while (x != root && colorOf(x) == BLACK) {
// 删除节点是左子节点
if (x == leftOf(parentOf(x))) {
// sib是删除节点父节点的右子节点(兄弟节点)
Entry<K,V> sib = rightOf(parentOf(x));
// 兄弟节点是红色
if (colorOf(sib) == RED) {
// 情况1.1处理
setColor(sib, BLACK);
setColor(parentOf(x), RED);
rotateLeft(parentOf(x));
sib = rightOf(parentOf(x));
}
// sib兄弟节点有两个黑色的子节点,情况2处理
if (colorOf(leftOf(sib)) == BLACK &&
colorOf(rightOf(sib)) == BLACK) {
// 变色
setColor(sib, RED);
// 指针指向删除节点的父节点
x = parentOf(x);
} else {
// 兄弟节点的右子节点是黑色
if (colorOf(rightOf(sib)) == BLACK) {
// 情况3.1.1处理
setColor(leftOf(sib), BLACK);
setColor(sib, RED);
rotateRight(sib);
sib = rightOf(parentOf(x));
}
// 情况3.1.2处理
setColor(sib, colorOf(parentOf(x)));
setColor(parentOf(x), BLACK);
setColor(rightOf(sib), BLACK);
rotateLeft(parentOf(x));
// 跳出循环
x = root;
}
} else { // symmetric
// 删除节点是右子节点
// sib是删除节点父节点的左子节点(兄弟节点)
Entry<K,V> sib = leftOf(parentOf(x));
// 兄弟节点是红色
if (colorOf(sib) == RED) {
// 情况1.2处理
setColor(sib, BLACK);
setColor(parentOf(x), RED);
rotateRight(parentOf(x));
sib = leftOf(parentOf(x));
}
// sib兄弟节点有两个黑色的子节点,情况2处理
if (colorOf(rightOf(sib)) == BLACK &&
colorOf(leftOf(sib)) == BLACK) {
// 变色
setColor(sib, RED);
// 指针指向删除节点的父节点
x = parentOf(x);
} else {
// 兄弟节点的左子节点是黑色
if (colorOf(leftOf(sib)) == BLACK) {
// 情况3.2.1处理
setColor(rightOf(sib), BLACK);
setColor(sib, RED);
rotateLeft(sib);
sib = leftOf(parentOf(x));
}
// 情况3.2.2处理
setColor(sib, colorOf(parentOf(x)));
setColor(parentOf(x), BLACK);
setColor(leftOf(sib), BLACK);
rotateRight(parentOf(x));
// 跳出循环
x = root;
}
}
} setColor(x, BLACK);
}

参考

结语

欢迎关注微信公众号『码仔zonE』,专注于分享Java、云计算相关内容,包括SpringBoot、SpringCloud、微服务、Docker、Kubernetes、Python等领域相关技术干货,期待与您相遇!

彻底理解红黑树及JavaJDK1.8TreeMap源码分析的更多相关文章

  1. 《深入理解Spark:核心思想与源码分析》——SparkContext的初始化(叔篇)——TaskScheduler的启动

    <深入理解Spark:核心思想与源码分析>一书前言的内容请看链接<深入理解SPARK:核心思想与源码分析>一书正式出版上市 <深入理解Spark:核心思想与源码分析> ...

  2. 《深入理解Spark:核心思想与源码分析》(前言及第1章)

    自己牺牲了7个月的周末和下班空闲时间,通过研究Spark源码和原理,总结整理的<深入理解Spark:核心思想与源码分析>一书现在已经正式出版上市,目前亚马逊.京东.当当.天猫等网站均有销售 ...

  3. 《深入理解Spark:核心思想与源码分析》(第2章)

    <深入理解Spark:核心思想与源码分析>一书前言的内容请看链接<深入理解SPARK:核心思想与源码分析>一书正式出版上市 <深入理解Spark:核心思想与源码分析> ...

  4. 《深入理解Spark:核心思想与源码分析》一书正式出版上市

    自己牺牲了7个月的周末和下班空闲时间,通过研究Spark源码和原理,总结整理的<深入理解Spark:核心思想与源码分析>一书现在已经正式出版上市,目前亚马逊.京东.当当.天猫等网站均有销售 ...

  5. 《深入理解Spark:核心思想与源码分析》正式出版上市

    自己牺牲了7个月的周末和下班空闲时间,通过研究Spark源码和原理,总结整理的<深入理解Spark:核心思想与源码分析>一书现在已经正式出版上市,目前亚马逊.京东.当当.天猫等网站均有销售 ...

  6. 深入理解分布式调度框架TBSchedule及源码分析

    简介 由于最近工作比较忙,前前后后花了两个月的时间把TBSchedule的源码翻了个底朝天.关于TBSchedule的使用,网上也有很多参考资料,这里不做过多的阐述.本文着重介绍TBSchedule的 ...

  7. TreeMap源码分析,看了都说好

    概述 TreeMap也是Map接口的实现类,它最大的特点是迭代有序,默认是按照key值升序迭代(当然也可以设置成降序).在前面的文章中讲过LinkedHashMap也是迭代有序的,不过是按插入顺序或访 ...

  8. Java——HashMap底层源码分析

    1.简介 HashMap 根据键的 hashCode 值存储数据,大多数情况下可以直接定位到它的值,因而具有很快的访问速度,但遍历顺序却是不确定的. HashMap 最多只允许一条记录的key为 nu ...

  9. [JUC-5]ConcurrentHashMap源码分析JDK8

    在学习之前,最好先了解下如下知识: 1.ReentrantLock的实现和原理. 2.Synchronized的实现和原理. 3.硬件对并发支持的CAS操作及JVM中Unsafe对CAS的实现. 4. ...

随机推荐

  1. Go测试开发(一) 怎么写Go代码

    安装过程略过,网上搜一大把. 介绍 本文会在一个module中开发一个简单的Go package. 同时介绍go tool(也就是go命令行). 以及如何fetch,build和install Go的 ...

  2. Diophantus of Alexandria(唯一分解定理)

    Diophantus of Alexandria was an Egypt mathematician living in Alexandria. He was one of the first ma ...

  3. 02 axios

    request.js import axios from 'axios' const config = require('@/config') const instance = axios.creat ...

  4. 【Gin-API系列】部署和监控(九)

    本文是[Gin-API系列]的最后一篇文章,简单介绍如何在生产环境的部署架构和监控手段. 生产部署 部署架构 使用Nginx加Keepalived的方式搭建,可以达到高可用的效果,并可以横向扩容 如何 ...

  5. SpringBean容器启动流程+Bean的生命周期【附源码】

    如果对SpringIoc与Aop的源码感兴趣,可以访问参考:https://javadoop.com/,十分详细. 目录 Spring容器的启动全流程 Spring容器关闭流程 Bean 的生命周期 ...

  6. 执行Python程序出现“SyntaxError: Non-UTF-8 code starting with '\xb6'...”错误怎么办?

    如果文件中有中文,直接执行python xx.py会出现以下错误: SyntaxError: Non-UTF- code starting with '\xb6' in file XX.py on l ...

  7. Zookeeper高级

    1.1. 一致性协议概述 前面已经讨论过,在分布式环境下,有很多不确定性因素,故障随时都回发生,也讲了CAP理论,BASE理论 我们希望达到,在分布式环境下能搭建一个高可用的,且数据高一致性的服务,目 ...

  8. Python实现加密的RAR文件解压(密码已知)

    博主之前在网上找了很多资料,发现rarfile库不能直接调用,需要安装unrar模块,下面将详细介绍整个实现流程. 第一步:安装unrar模块,直接pip install unrar可能会找不到库,需 ...

  9. Javaweb项目重新编译方式

    project->clean选中需要重新编译的项目确定即可 我不知道重新编译的说法是否准确,但是我喜欢这么叫

  10. python中faker模块:产生随机数据的模块

    #pip install faker #产生各种随机数据的模块 想要运用更多的随机数据,可以百度查找下