带你认识红黑树

红黑树

  • 一、什么是红黑树?
    • 1.1 AVL树
    • 1.2 红黑树
  • 二、红黑树的特点
  • 三、红黑树的insert、delete
    • 3.1 insert
      • 3.1.1 父节点为空
      • 3.1.2 父节点为Black节点
      • 3.1.3 父节点为Red节点
        • 3.1.3.1 叔叔节点为Red节点
        • 3.1.3.2 叔叔节点为Black节点
    • 3.2 delete
    • 3.2.1 删除节点有两个子节点
    • 3.2.2 删除节点只有一个子节点
    • 3.2.3 删除节点无子节点
      • 3.2.3.1 删除节点为红色节点
      • 3.2.3.2 删除节点为黑色节点
        • 1. 兄弟节点为Red
        • 2. 兄弟节点为黑色节点

一、什么是红黑树?

在谈及红黑树前,首先我们先了解一下什么AVL(平衡二叉树)树。

1.1 AVL树

AVL树,也叫平衡二叉树,具有以下特点:

  1. 本身就为一颗 二叉搜索树
  2. 每个结点的左右子树的高度之差的绝对值最多为 1。
  3. 每个节点都是 AVL 树。
  4. 可以为空。
    在这里插入图片描述
    AVL树相比于二叉搜索树,具有更高的查询效率,因为在一些极端场景下,二叉搜索树(Binary Search Tree),会蜕化为链表(如下图),使得搜索的时间复杂度为 O(N),AVL树时间复杂度严格为 O(log2N)。
    在这里插入图片描述

因为具有以上特点,所以在进行插入、删除节点时,需通过多次旋转节点来满足上述特点,因此,AVL树在查找上效率较高,但是在插入、删除上,效率较低。

因此,一些大牛为满足效率上的提升,对AVL树进行“升级”,红黑树概念由此产生,具体如下:

1.2 红黑树

红黑树是一种特殊的AVL树,通过在插入、删除时多次旋转节点来保持二叉查找树的平衡,从而获得较高的查找性能(O(log2N))。但是红黑树又是一种非严格意义上的AVL树,它的左右子树高差有可能大于 1,但对之进行平衡的代价较低, 使得其性能要远远强于 AVL树。所以在多数计算机语言中,通用使用红黑树,来替代AVL树,进行高效查找、删除、插入。

在这里插入图片描述

二、红黑树的特点

红黑树是一种自平衡的AVL树,其特点如下:

  1. 可以为空树。
  2. 如果不为空树,其根节点必须为黑色节点。
  3. 每个节点的左子节点值都小于父节点,每个节点右子节点都大于父节点值。
  4. 任何相邻的节点都不能同时为红色节点。
  5. 每个叶子节点都具有黑色的空节点(NIL),如下图,叶子节点不存储数据,只为满足特点6。
  6. 每个节点,从该节点到所有可达叶子节点的所有路径,其包含的黑色节点数目都相同,如节点 0004到所有叶子节点都有两个黑色节点。
  7. 节点的插入,都为红色节点,通过旋转着色来满足红黑树的其他特征。
    在这里插入图片描述

三、红黑树的insert、delete

3.1 insert

红黑树的节点insert,初始颜色都为Red,通过旋转着色来满足红黑树的其他特征。节点的insert又可以分为以下几种情况:

  1. 插入的节点,其父节点为空;
  2. 插入的节点,其父节点为Black节点;
  3. 插入的节点,其父节点为Red节点;
    3.1 叔叔节点为Black节点;
    3.2 叔叔节点为Red节点;

具体详情下面一一分析。

3.1.1 父节点为空

父节点为空,既当前Tree 为空,直接插入。

通过红黑树的特点可知,根节点为Black节点,所以需要重新着色。

3.1.2 父节点为Black节点

父节点为Black节点,插入节点为Red节点,由红黑树特点可知,直接插入当前节点即可满足其所有特点。

3.1.3 父节点为Red节点

当父节点为Red节点时,不满足红黑树特点:任何相邻的节点都不能同时为红色节点,所以需要旋转、着色。

3.1.3.1 叔叔节点为Red节点

在这里插入图片描述
如上图,当插入节点父节点、叔叔节点均为Red节点时,为满足特点:任何相邻的节点都不能同时为红色节点;每个节点,从该节点到所有可达叶子节点的所有路径,其包含的黑色节点数目都相同。所以,需要重新进行着色,将父节点、叔叔节点颜色修改为Black,爷爷节点修改为Red节点。
在这里插入图片描述
这就完了么???NO!NO!NO!
试想一下,如果祖父节点恰好为Red节点,是否相邻节点又都为红色节点,如下图所示:
在这里插入图片描述
插入一个新节点 800,满足父节点、叔叔节点均为红色,则先将父节点、叔叔节点颜色修改为Black,爷爷节点修改为Red节点。如下:
在这里插入图片描述
这时,节点 0200 与 节点 0300 都为Red节点,且 0300 的兄弟节点一定为Black节点(任何相邻的两个节点不能同时为Red节点),所以将 0300 节点当做新插入的节点,情况就变为了 3.1.3.2 叔叔节点为Black节点,具体操作步骤下面详细分析。

3.1.3.2 叔叔节点为Black节点

当父节点为Red节点、叔叔节点为Black节点,如下图:
在这里插入图片描述
插入节点 0500,其父节点为Red节点,叔叔节点为NIL黑色节点,需根据具体情况来进行旋转、着色,如下:
condition 1(左左):父节点为左子节点,插入节点为左子节点。

在这里插入图片描述
condition 2(左右):父节点为左子节点,插入节点为右子节点。
在这里插入图片描述

condition 3(右右):父节点为右子节点,插入节点为右子节点。
在这里插入图片描述

condition 4(右左):父节点为右子节点,插入节点为左子节点。
在这里插入图片描述

3.2 delete

红黑树节点的删除与二叉搜索树(AVL树)一样,也分为三种情况,如下:

  1. 删除节点无子节点;
  2. 删除节点只有一个子节点;
  3. 删除节点有两个子节点;

红黑树删除,采用以繁化简进行处理,**对于非叶子节点的删除,最终都转化为对叶子节点的删除。**通过这个思想,下面对以上情况进行详细分析。

3.2.1 删除节点有两个子节点

在了解中间节点的删除前,我们先了解两个概念:

  • 前驱节点: 在中序遍历中,一个节点的前驱结点,先找到该节点的左孩子节点,再找左孩子节点的最后一个右孩子节点。向左走一步,然后向右走到头。最后一个右孩子节点即为前驱节点。

  • 后继节点: 在中序遍历中,一个节点的后继结点,先找到该节点的右孩子节点,再找右孩子节点的最后一个左孩子节点。向右走一步,然后向左走到头。最后一个左孩子节点即为后继结点。

采用以繁化简进行处理,对于非叶子节点的删除,最终都转化为对叶子节点的删除。

  1. 当删除节点为中间节点是,寻找当前节点的前驱节点,交换前驱节点与删除节点的值,然后直接删除前驱节点;
  2. 如果不存在前驱节点,则寻找当前接的后继节点,交换后继节点与删除节点的值,直接删除后继节点;

通过上述操作,将复杂的中间节点删除,转换为叶子节点删除,叶子节点的删除,参照步骤 3.2.2 删除节点只有一个子节点、3.2.3 删除节点无子节点

3.2.2 删除节点只有一个子节点

将删除节点的父节点指向当前节点的子节点,直接删除,并交换删除节点、子节点颜色,保持颜色平衡,如下:

在这里插入图片描述

  • 将删除节点的父节点指向当前节点的子节点
  • 直接删除当前节点
    在这里插入图片描述
  • 交换删除节点、子节点颜色
    在这里插入图片描述

3.2.3 删除节点无子节点

3.2.3.1 删除节点为红色节点

当前模式最为简单,红色叶子节点删除,不影响红黑树平衡,所以可以直接删除即可。

在这里插入图片描述

  • 删除红色节点 250:直接删除
    在这里插入图片描述

3.2.3.2 删除节点为黑色节点

当删除节点为黑色节点时,由于影响到红黑树的特点(每个节点,从该节点到所有可达叶子节点的所有路径,其包含的黑色节点数目都相同,这种特定称为黑高相同),所以需要进行旋转、着色来修复红黑树。

可分为以下几种情况。

1. 兄弟节点为Red

  • 1 父节点为红色节点:不存在当前情况,无需考虑。
  • 2 父节点为黑色节点;
    • 2.1 兄弟节点无子节点:不存在当前情况,需保持黑高相同
    • 2.2 兄弟节点存在一个子节点:不存在当前情况,需保持黑高相同
    • 2.3 兄弟节点存在两个子节点:分析可知,两个子节点均为红色节点

最终,当兄弟节点为Red节点时,父节点一定为Black节点,且兄弟节点一定存在两个子节点,两个子节点均为红色,如下:
在这里插入图片描述
(图1)

删除步骤如下:

  • 直接删除节点;
  • 将兄弟节点设置为Black;
  • 如果删除节点为左子节点
    • 则将兄弟节点左子节点设置为红色;
    • 对父节点进行左旋
    • 如果兄弟节点的左子节点(BLeft)存在子节点;
      • 存在左子节点:对BLeft进行右旋,然后对BLeft的父节点进行左旋,设置BRight左子节点为Black。
      • 存在右子节点:对BLeft的父节点进行左旋,交换BLeft与BLeft右子节点的颜色。
      • 存在左右子节点:对BLeft的父节点进行左旋,交换BLeft与BLeft右子节点的颜色。
  • 如果删除节点为右子节点
    • 则将兄弟节点右子节点设置为红色;
    • 对父节点进行右旋
    • 如果兄弟节点的右子节点(BRight)存在子节点;
      • 存在左子节点:对BRight的父节点进行右旋,交换BRight与BRight左子节点的颜色。
      • 存在右子节点:对BRight进行左旋,然后对BRight的父节点进行右旋,设置BRight右子节点为Black。
      • 存在左右子节点:对BRight的父节点进行右旋,交换BRight与BRight左子节点的颜色。

删除图1中的Black节点 0050,通过以上步骤,最终红黑树如图二所示。
图二
(图二)

2. 兄弟节点为黑色节点

注意:分析可知,当前节点为Black节点,兄弟节点也为Black节点,则如果兄弟节点存在子节点,子节点颜色一定为Red节点。

  • 1 父节点为红色节点:
    • 1.1 兄弟节点无子节点:直接交换父节点、兄弟节点颜色;
    • 1.2 兄弟节点存在一个子节点;
      • 1.2.1 存在左子节点:① 删除节点为左子节点:将兄弟节点右旋,然后将父节点左旋,最后把兄弟节点左子节点颜色修改为Black,并修改兄弟节点颜色为Red;② 删除节点为右子节点:将父节点右旋;
      • 1.2.2 存在右子节点:① 删除节点为左子节点:将父节点左旋;② 删除节点为右子节点:将兄弟节点左旋,然后将父节点右旋,最后把兄弟节点右子节点颜色修改为Black,并修改兄弟节点颜色为Red;
    • 1.3 兄弟节点存在两个子节点;
      • 1.3.1 删除节点为左子节点:将父节点左旋,然后将兄弟节点右子节点修改为Black;
      • 1.3.2 删除节点为右子节点:将父节点右旋,右旋后,会导致两个节点连续为红色,把之前兄弟节点的右子节点当做插入的节点,走新增节点3.1.3父节点为Red节点、3.1.3.1叔叔节点为Red节点;
  • 2 父节点为黑色节点;
    • 2.1 兄弟节点无子节点;
      • 2.1.1 不存在祖父节点:直接将兄弟节点颜色修改为红色;
      • 2.1.2 存在祖父节点:直接将兄弟节点颜色修改为红色;以父节点作为条件判断,将父节点赋值个当前节点,递归处理,判断其父节点、兄弟节点颜色。
    • 2.2 兄弟节点存在一个子节点;
      • 2.2.1 存在左子节点:① 删除节点为左子节点:将兄弟节点右旋,然后将父节点左旋,最后把兄弟节点左子节点颜色修改为Black;② 删除节点为右子节点:将父节点右旋,然后把兄弟节点左子节点颜色修改为Black;
      • 2.2.2 存在右子节点:① 删除节点为左子节点:将父节点左旋,然后把兄弟节点右子节点颜色修改为Black;② 删除节点为右子节点:将兄弟节点左旋,然后将父节点右旋,最后把兄弟节点右子节点颜色修改为Black;
    • 2.3 兄弟节点存在两个子节点;
      • 2.3.1 删除节点为左子节点:将父节点左旋,然后将兄弟节点右子节点修改为Black;
      • 2.3.2 删除节点为右子节点:将父节点右旋,然后将兄弟节点左子节点修改为Black;

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/65696.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Scratch 之 TurboWarp 常用插件介绍-1

今天带来2篇 TurboWarp 常用插件介绍。 什么你还没有 TurboWarp ?快去下载一个吧 TurboWarp(简称TW) 在线版 | 离线版下载 TurboWarp优点 编译速度快于原版 Scratch 至少10倍拥有自定义帧的功能(比如60 FPS)造型编…

【博客691】VictoriaMetrics如何支持Multi Retention

VictoriaMetrics如何支持Multi Retention 场景: 实现Multi Retention Setup within VictoriaMetrics Cluster,使得为不同的监控数据采用不同的保存时间 Multi Retention实现方式 方式: VictoriaMetrics 的社区版本通过 -retentionPeriod 命…

【工具插件类教学】电脑端移动端缩放大图自适应Simple Zoom

目录 简介 1.创建Canvas并设置 2.使用预制体Zoom 3.商店地址 简介 特点: •易于使用和高度可定制。 •支持鼠标(桌面)和触摸(移动)。 •指定最小和最大缩放的限制。 •缩放指针(鼠标/手指)或屏幕上预定义的自定义位置。 •变焦时使用夹紧/弹性变焦类型。 •定义缩…

基于PHP的轻量级博客typecho

本文完成于 5 月中旬,发布时未在最新版本上验证; 什么是 typecho ? Typecho 是一款基于 PHP 的博客软件,旨在成为世界上最强大的博客引擎。Typecho 在 GNU 通用公共许可证 2.0 下发布。支持多种数据库,原生支持 Markdo…

征稿 | 第三届粤港澳大湾区人工智能与大数据论坛(AIBDF 2023)

第三届粤港澳大湾区人工智能与大数据论坛(AIBDF 2023) 2023 3rd Guangdong-Hong Kong-Macao Greater Bay Area Artificial Intelligence And Big Data Forum 本次高端论坛围绕建设国家数字经济创新发展试验区进行选题。全面贯彻落实党的二十大精神&…

【C++进阶之路】继承与多态的概念考察

文章目录 一、问答题二、概念题三、答案与解析问答题概念题 一、问答题 什么是菱形继承?菱形继承的问题是什么?什么是菱形虚拟继承?如何解决数据冗余和二义性的。继承和组合的区别?什么时候用继承?什么时候用组合&…

linux基于信号量实现多线程生产者消费者模型

基于信号量实现多线程生产者消费者模型。 编程思路: 1.食物的初始化编号为100: beginnum 100; 2.仓库有5个空碗,最多保存5个食物:queue[5]; 3.初始化空碗的数量为5,食物的数量为0&#xff1a…

FFmpeg中AVIOContext的使用

通过FFmpeg对视频进行编解码时,如果输入文件存在本机或通过USB摄像头、笔记本内置摄像头获取数据时,可通过avformat_open_input接口中的第二个参数直接指定即可。但如果待处理的视频数据存在于内存块中时,该如何指定,可通过FFmpeg…

《孙子兵法》快速概览,有哪些章节?趣讲《孙子兵法》【第2讲】

《孙子兵法》快速概览,有哪些章节?趣讲《孙子兵法》【第2讲】 《孙子兵法》十一家注是一个有名的版本,十一家注是曹操、杜牧等十一人注释,曹操是真正的军事家,是名副其实的大咖。总共三卷十三篇,比较难记住…

3个月快速入门LoRa物联网传感器开发

在这里插入图片描述 快速入门LoRa物联网传感器开发 LoRa作为一种LPWAN(低功耗广域网络)无线通信技术,非常适合物联网传感器和行业应用。要快速掌握LoRa开发,需要系统学习理论知识,并通过实际项目积累经验。 摘要: 先学习LoRa基础知识:原理、网络架构、协议等,大概需要2周时间…

Java加密算法的应用与实现(MD5、SHA、DES、3DES、AES、RSA、ECC)

文章目录 一、散列加密算法1、概述2、常见算法(MD5、SHA)3、应用4、Java实现 二、对称加密算法1、概述2、常见算法(DES、3DES、AES)3、应用4、Java实现AES 三、非对称加密算法1、概述2、常见算法(RSA、ElGamal、Rabin、…

【linux】ssh 和adb connect区别

问:ssh 与ping的区别 答:SSH(Secure Shell)和Ping是两种完全不同的网络工具。 SSH是一种加密的网络协议,用于安全地远程管理或访问远程计算机。它提供了一种安全的通信方式,可以在不安全的网络上进行远程登…

淘宝API接口为开发者提供了与淘宝平台进行数据交互和操作的便捷途径

淘宝API接口是指淘宝开放平台提供的一套接口,用于与淘宝网进行数据交互和操作。通过使用淘宝API接口,第三方开发者可以实现商品搜索、店铺信息获取、订单管理、商家服务等功能,从而实现与淘宝平台的对接和数据共享。 淘宝API接口的使用可以帮…

备忘录模式(Memento)

备忘录模式是一种行为设计模式,在不破坏封装性的前提下,允许在不暴露对象实现细节的情况下保存和恢复对象之前的状态。 Memento is a behavior design pattern. Without compromising encapsulation, it can reserve and restore of the previous stat…

vscode自动添加注释说明

1. 安装vscode 双击安装程序,默认安装即可(如:VSCodeSetup-x64-1.70.2.exe) 2. 安装doxygen文档生成插件 1> 打开vscode软件,点击左侧插件管理菜单 2> 点击右上角’…‘按钮,选择’Install from VSIX’(联网状态可以直接搜索doxygen下载安装) 3> 选择doxygen离线安装…

Java课题笔记~ 使用 Spring 的事务注解管理事务(掌握)

通过Transactional 注解方式,可将事务织入到相应 public 方法中,实现事务管理。 Transactional 的所有可选属性如下所示: propagation:用于设置事务传播属性。该属性类型为 Propagation 枚举, 默认值为 Propagation.R…

【SpringBoot】日志是什么+基于lombok的日志输出

博主简介:想进大厂的打工人博主主页:xyk:所属专栏: JavaEE进阶 在我们日常的程序开发中,日志是程序的重要组成部分,想象⼀下,如果程序报错了,不让你打开控制台看⽇志,那么你能找到报错的原因吗…

BEM命名规范

参加了一个团队开发的小项目,代码写完了一看别人的感觉自己写的老不规范了,后知后觉才看到开发文档里面的样式书写规范。感觉要大改了……也算给自己长个记性要先读完所有文档在开始。 也学习了解了一下BEM命名规范。 1. 什么是BEM? BEM&a…

软件测试(功能、接口、性能、自动化)详解

一、软件测试功能测试 测试用例编写是软件测试的基本技能;也有很多人认为测试用例是软件测试的核心;软件测试中最重要的是设计和生成有效的测试用例;测试用例是测试工作的指导,是软件测试的必须遵守的准则。 黑盒测试常见测试用…

MinIO:微服务中上传图片流程

1、在nacos中配置minio参数 2、controller层 package com.heima.wemedia.controller.v1;import com.heima.model.common.dtos.ResponseResult; import com.heima.wemedia.service.WmMaterialService; import org.springframework.beans.factory.annotation.Autowired; import …