带你认识红黑树

news2025/1/11 7:01:21

红黑树

  • 一、什么是红黑树?
    • 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.coloradmin.cn/o/851082.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

libmpv使用滤镜处理视频进行播放

一、前言 作为一个功能强大的多媒体框架,libmpv为开发者提供了广泛的功能和灵活的控制权。滤镜是libmpv的一个重要特性,允许开发者对视频进行各种实时处理和增强,从而满足用户对于个性化、创意化和高质量视频体验的需求。 滤镜是一种在视频渲染过程中应用特定效果的技术。…

若依-plus-vue启动显示Redis连接错误

用的Redis是windows版本,6.2.6 报错的主要信息如下: Failed to instantiate [org.redisson.api.RedissonClient]: Factory method redisson threw exception; nested exception is org.redisson.client.RedisConnectionException: Unable to connect t…

Vscode-工具使用

Vscode ,这玩意儿是开源的,以前用收费的破解版,过段时间就高版本不匹配,这次搞个不要钱的玩玩,记录使用心得 下载 下载地址:官网 点击下载,但是这里有个问题下载比较慢,解决办法&a…

Redhat Linux 安装MySQL安装手册

Redhat安装MySQL安装手册 1 下载2 上传服务器、解压并安装3 安装安装过程1:MySQL-shared-5.6.51-1.el7.x86_64.rpm安装过程2:MySQL-shared-compat-5.6.51-1.el7.x86_64.rpm安装过程3:MySQL-server-5.6.51-1.el7.x86_64.rpm安装过程4&#xff…

从碎片化到整体性,医美服务的下一个战场

近年来,医美消费需求蓬勃增长,各类医美机构、医美平台如雨后春笋般涌现,为医美消费者提供了更多选择的同时,也赋予了他们更大的权利。据调查,在经历了一次不愉快的体验之后,“换一家”——是绝大多数医美消…

k8s --pod详解

目录 一、Pod基础概念 1、pod简介 2、在Kubrenetes集群中Pod有如下两种使用方式 3、pause容器使得Pod中的所有容器可以共享两种资源:网络和存储。 (1)网络 (2)存储 4、kubernetes中的pause容器主要为每个容器提供…

IELTS图表类作文基础知识

表格可以用table或chart来表示。 其实,数据类图表除了可以用chart表示,也可以用其他单词。 表格又可以称为table。而带有几何图形的图表可以用graph来表示。 像饼状图、折线图、柱状图这样用几何图形,或者直接用表格来呈现数据的形式&#x…

macOS(m芯片)连接服务器及其进行文件传输的各种方式的详解

说明:使用了macOS后发现,win系统能使用的xshell、xftp等连接服务器及其文件传输等软件均不能使用了,没有兼容的版本。所以我们刚切换到mac系统该如何去适应呢。 一、连接远程服务器 macOS中前文也说道我们使用的是iterm2进行终端控制的&…

【高级程序设计语言C++】红黑树

1. 红黑树的概念2. 红黑树的插入2.1. 情况12.2. 情况22.3. 情况32.4. 插入情况小总结 3. 红黑树与AVL树的对比4. 红黑树在线生成网站 1. 红黑树的概念 红黑树(Red-Black Tree)是一种自平衡的二叉搜索树,它在插入和删除操作时通过调整节点的颜…

anylabeling安装与使用说明

文章目录 一.anylabeling说明二. 安装教程1. 可执行程序方式2. python程序 一.anylabeling说明 官网:https://anylabeling.nrl.ai/docs 该工具作为一个具有Segment Anything和YOLO模型的智能标签工具,可以快速、准确地对图像进行标注。 二. 安装教程 1. 可执行程…

Docker Desktop 启用 Kubernetes 失败后处理

一、环境 Windows 10 C:\Users\zhuji>docker --version Docker version 24.0.2, build cb74dfc 二、问题 在setting -> Kubernetes 中,选中 Enable Kubernetes 后,长时间显示 Starting ... ,在Images中显示几个自动下载的镜像后&…

e6zzseo:外贸独立站怎么推广

外贸独立站的推广需要一系列综合性的策略和方法,以吸引目标市场的访问者,并将他们转化为潜在客户。以下是一些推广外贸独立站的建议: 1. 搜索引擎优化(SEO): e6zzseo认为优化网站可以适应搜索引擎的要求&a…

1749. 任意子数组和的绝对值的最大值

诸神缄默不语-个人CSDN博文目录 力扣刷题笔记 文章目录 1. 暴力搜索2. 动态规划3. 前缀和 1. 暴力搜索 直接用2个指针从索引0开始找到最后一个索引,时间复杂度大概是 O ( n 2 ) O(n^2) O(n2)吧,总之这么搞不行,以下是我用Python写的一些典型…

【2.1】Java微服务: Nacos注册中心

目录 Nacos介绍 Nacos安装 下载和安装 修改端口 启动 服务注册与发现 导入Nacos管理依赖 导入服务依赖 配置Nacos的服务地址 启动服务,查看已注册的服务 服务分级存储模型 分级存储模型介绍 具体结构 配置实例集群 同集群优先的负载均衡策略 服务权重配置…

vue中vuex的五个属性和基本用法,另加js-cookie的使用

VueX 是一个专门为 Vue.js 应用设计的状态管理构架,统一管理和维护各个vue组件的可变化状态(你可以理解成 vue 组件里的某些 data )。 Vuex有五个核心概念: state, getters, mutations, actions, modules。 1. state: vuex的基本数据&…

【力扣刷题 | 第二十五天】

目录 前言: 474. 一和零 - 力扣(LeetCode) 总结: 前言: 今天我们依旧暴打动态规划 474. 一和零 - 力扣(LeetCode) 给你一个二进制字符串数组 strs 和两个整数 m 和 n 。 请你找出并返回 strs 的最大子集…

docker下载和案例

文章目录 Docker安装一,根据官方文档安装二,根据我以下方式 Docker配置错误导致漏洞一,CRLF注入漏洞介绍在nginx中该漏洞例子解决方法 目录穿越漏洞介绍解决方法 Docker安装 一,根据官方文档安装 官方文档 二,根据我以下方式 docker安装要求: Docker要求Ce…

Unity游戏源码分享-植物大战僵尸素材与源码

Unity游戏源码分享-植物大战僵尸素材与源码 完整版本下载地址: https://download.csdn.net/download/Highning0007/88191862

【Windbg】通过网络调试windows内核

环境 windows版本:win10_x64 1901 windbg版本:1.2306.12001.0 HOST 1、windbg软件设置。 点击菜单文件,然后如下图操作。 2、等待连接。 ************* Waiting for Debugger Extensions Gallery to Initialize **************>>&…

string模拟实现:

string模拟实现: 上一篇博客,我们对String类有了一个基本的认识,本篇博客我们来从0~1去模拟实现一个String类,当然我们实现的都是一些常用的接口。 ❓我们这里定义了一个string类型,然后STL标准库里面也有string&#…