java面试-彻底搞懂红黑树

简介: 红黑树性质1、每个结点或是红色的,或是黑色的 2、根节点是黑色的 3、每个叶结点(NIL)是黑色的 4、如果一个节点是红色的,则它的两个儿子都是黑色的。

红黑树性质

1、每个结点或是红色的,或是黑色的 
2、根节点是黑色的 
3、每个叶结点(NIL)是黑色的 
4、如果一个节点是红色的,则它的两个儿子都是黑色的。 
5、对于每个结点,从该结点到其叶子结点构成的所有路径上的黑结点个数相同。

和AVL树的比较

AVL树是一棵严格的平衡树,它所有的子树都满足二叉平衡树的定义。因此AVL树高被严格控制在XXX,因此AVL树的查找比较高效。但AVL树插入、删除结点后旋转的次数比红黑树多。

红黑树用非严格的平衡来降低插入删除时旋转的次数。

因此,如果你的业务中查找远远多于插入、删除,那选AVL树; 
如果查找、插入、删除频率差不多,那么选择红黑树。

插入过程

默认插入的结点为红色。为何? 
因为红黑树中黑节点至少是红节点的两倍,因此插入节点的父节点为黑色的概率较大,而此时并不需要作任何调整,因此效率较高。

1. 父为黑

title 
插入后无需任何操作。由于黑节点个数至少为红节点的两倍,因此父为黑的情况较多,而这种情况在插入后无需任何调整,这就是红黑树比AVL树插入效率高的原因!

2. 父为红

父为红的情况破坏了红黑树的性质,此时需要根据叔叔的颜色来做不同的处理。 
title

  1. 叔叔为红 
    title 
    此时很简单,只需交换爸爸、叔叔和爷爷的颜色即可。 
    此时若爷爷节点和太爷爷节点颜色相同,再以爷爷节点为起始节点,进行刚才相同的操作,即:根据爷爷的兄弟颜色做相应的操作。

  2. 叔叔为黑 
    此时较为复杂,分如下四种情况: 
    a)爸爸在左、叔叔在右、我在左 
    title 
    以爸爸为根节点,进行一次R旋转。 
    b)爸爸在左、叔叔在右、我在右 
    title 
    title 
    先以我为根节点,进行一次L旋转; 
    再以我为根节点,进行一次R旋转。 
    c)叔叔在左、爸爸在右、我在左 
    title 
    先以我为根节点,进行一次R旋转; 
    再以我为根节点,进行一次L旋转。 
    d)叔叔在左、爸爸在右、我在右 
    title 
    以爸爸为根节点,进行一次L旋转。


删除过程

二叉搜索树的删除

若删除二叉搜索树的节点A,实际上删除的是二叉搜索树中序遍历的前驱节点,注意: 
1. 这个被删除节点要么就是一个叶子节点, 
2. 要么有且仅有一个左孩子 
然后将孩子顶替它原来的位置,最后将被删的节点值覆盖待删除的那个节点A。

红黑树按照二叉搜索树的方式删除节点,之后再进行相应的旋转操作,使得删除后的树仍然是一棵红黑树。

定义

  1. 待删除节点:要删除的那个节点
  2. 实际删除节点:待删除节点的中序遍历前驱

红黑树实际删除节点的性质

  1. 实际删除节点要么是叶子节点,要么有且仅有一个左孩子;
  2. 若为叶子节点,必为红色;
  3. 若实际删除节点还有孩子,则该必为左孩子; 
    a)若左孩子为红色,则实际删除节点必为黑色; 
    b)若左孩子为黑色,则实际删除节点红黑均可以。

约定

  1. 蓝色箭头:表示判定点
  2. 在删除操作开始前,蓝色箭头首先指向实际删除节点。
  3. 『实际删除节点』在图中以『父』表示。

旋转过程开始:

1. 父为红色(待删节点为叶子)

直接删除父节点即可: 
title


2. 父为黑 子为红(待删节点为黑、待删节点子节点为红+左孩子)

用子节点覆盖父节点,并保持父节点的颜色: 
title


3. 父为黑 子为黑(待删节点和子节点均为黑)

3.1. 叔叔为红

PS:叔叔为红,则爷爷必为黑!

  1. 父在左 叔在右 
    a)子节点覆盖父节点 
    b)进行一次左旋 
    title

  2. 父在右 叔在左 
    a)子节点覆盖父节点 
    b)进行一次右旋


3.2. 叔叔为黑

PS:叔叔、爸爸都为黑,那爷爷颜色就不确定了!

  1. 祖父红 两个侄子黑 
    以下两种情况操作一致: 
    1.子覆盖父(删除) 
    2.交换祖父和叔叔的颜色。

    a)父在左 叔在右 
    title 
    b)父在右 叔在左 
    同上。

  2. 祖父黑 两个侄子黑 
    以下两种情况操作一致: 
    1. 祖父染成子节点的颜色; 
    2. 子节点染成黑色; 
    3. 叔叔染成红色 
    a)父在左 叔在右 
    title 
    b)父在右 叔在左

  3. 祖父颜色随意 至少有一个红侄 
    a)红侄为左左(叔左、红侄左) 
    1. 红侄进行一次右旋 
    2. 红侄染成黑色 
    3. 交换叔叔和祖父的颜色 
    title 
    b)红侄为左右(叔左、红侄右) 
    1. 红侄进行一次右旋+左旋 
    2. 红侄染成父节点颜色; 
    3. 父节点染成黑色 
    title 
    c)红侄为右左(叔右、红侄左) 
    1. 红侄进行一次右旋+左旋 
    2. 红侄染成父节点颜色; 
    3. 父节点染成黑色; 
    title 
    d)红侄为右右(叔右、红侄右) 
    1. 红侄进行一次左旋 
    2. 叔叔染成父节点颜色; 
    3. 红侄染成黑色; 
    title

相关文章
|
29天前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
67 2
|
18天前
|
Java 程序员
Java社招面试题:& 和 && 的区别,HR的套路险些让我翻车!
小米,29岁程序员,分享了一次面试经历,详细解析了Java中&和&&的区别及应用场景,展示了扎实的基础知识和良好的应变能力,最终成功获得Offer。
45 14
|
28天前
|
存储 缓存 算法
面试官:单核 CPU 支持 Java 多线程吗?为什么?被问懵了!
本文介绍了多线程环境下的几个关键概念,包括时间片、超线程、上下文切换及其影响因素,以及线程调度的两种方式——抢占式调度和协同式调度。文章还讨论了减少上下文切换次数以提高多线程程序效率的方法,如无锁并发编程、使用CAS算法等,并提出了合理的线程数量配置策略,以平衡CPU利用率和线程切换开销。
面试官:单核 CPU 支持 Java 多线程吗?为什么?被问懵了!
|
1月前
|
存储 算法 Java
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
本文详解自旋锁的概念、优缺点、使用场景及Java实现。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
|
1月前
|
存储 缓存 Oracle
Java I/O流面试之道
NIO的出现在于提高IO的速度,它相比传统的输入/输出流速度更快。NIO通过管道Channel和缓冲器Buffer来处理数据,可以把管道当成一个矿藏,缓冲器就是矿藏里的卡车。程序通过管道里的缓冲器进行数据交互,而不直接处理数据。程序要么从缓冲器获取数据,要么输入数据到缓冲器。
Java I/O流面试之道
|
23天前
|
Java 编译器 程序员
Java面试高频题:用最优解法算出2乘以8!
本文探讨了面试中一个看似简单的数学问题——如何高效计算2×8。从直接使用乘法、位运算优化、编译器优化、加法实现到大整数场景下的处理,全面解析了不同方法的原理和适用场景,帮助读者深入理解计算效率优化的重要性。
27 6
|
1月前
|
存储 缓存 Java
大厂面试必看!Java基本数据类型和包装类的那些坑
本文介绍了Java中的基本数据类型和包装类,包括整数类型、浮点数类型、字符类型和布尔类型。详细讲解了每种类型的特性和应用场景,并探讨了包装类的引入原因、装箱与拆箱机制以及缓存机制。最后总结了面试中常见的相关考点,帮助读者更好地理解和应对面试中的问题。
53 4
|
1月前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
106 4
|
1月前
|
算法 Java
JAVA 二叉树面试题
JAVA 二叉树面试题
18 0
|
SQL 缓存 安全
Java高频面试题目
面试时面试官最常问的问题总结归纳!
148 0