红黑树 学习笔记
目录
1.红黑树的概念
1.1红黑树的规则
1.2红黑树的效率
2.红黑树的实现
2.1红黑树的大致结构
2.2红黑树的插入
2.2.1红黑树插入的大致过程
2.2.2情况1:变色
2.2.3情况2:单旋+变色
2.2.4情况3:双旋+变色
2.3红黑树的查找
1.红黑树的概念
红黑树是一颗搜索二叉树,相比于搜索二叉树,红黑树中的每个节点增加了一个存储位置来表示节点的颜色,可以是红色或黑色。通过对于任何一条从根到叶子的路径上各个节点的颜色进行约束,红黑树确保没有一条路径会比其他路径长出2倍,因此红黑树也算比较平衡的树。
1.1红黑树的规则
1.每个节点不是红色就是黑色。
2.根节点是黑色的
3.如果一个节点是红色的,那么该节点的两个孩子节点必须为黑色,也就是说红黑树中,任何路径都不会包含连续的红色节点。
4.对于任意一个节点,从该节点到其所有的NULL节点的简单路径上,均包含相同数量的黑色节点。
例如:

1.2红黑树的效率
假设N是红黑树中的节点数量,h为红黑树中最短路径的长度,对于 - 1 <= N <
-1, 由此可推出,红黑树增删查改最坏的情况也就是走
,那么时间复杂度也就为 O(
)。
例如:

2.红黑树的实现
2.1红黑树的大致结构
enum COLOUR
{RED,BLACK
};//COLOUR 来控制每个节点的颜色template<class K, class V>
struct RBTreeNode
{pair<K, V> _value;RBTreeNode<K, V>* _left;RBTreeNode<K, V>* _right;RBTreeNode<K, V>* _parent;COLOUR _col;// RED/BLACKRBTreeNode(const pair<K, V>& value):_value(value), _left(nullptr), _right(nullptr), _parent(nullptr),_col(RED) // 默认为红色,因为黑色会改变路径节点数量相等规则,较为麻烦{}
};template<class K, class V>
class RBTree
{using Node = RBTreeNode<K,V>;
public://... ...
private:Node* _root = nullptr;size_t num = 0;//记录节点数量
};
2.2红黑树的插入
2.2.1红黑树插入的大致过程
- 插入一个新节点时,按照二叉搜索树的规则进行插入,插入之后我们只需要判断是否符合红黑树的4条规则。
- 如果是空树插入,新增加的节点为黑色节点(也就是根节点)。如果是非空树插入,新增节点必须为红色节点。因为插入黑色节点,对于红黑树的每条路径黑色节点数量均相同的规则,我们很难处理。
- 非空树插入后,新增节点必须是红色的,如果父节点为黑色,则不违反规则,插入结束。
- 非空树插入后,若父节点为红色节点,则违反规则,因为不能出现连续的红色节点。进一步分析为以下几种情况:
2.2.2情况1:变色

例如上述的一种情况,x节点是我们的新插入节点,很明显,x和6构成了连续的红色节点,并且此时的g为黑,u存在且为红。我们如果想要红黑树符合规则,必须将6的颜色变为黑色,但是在18->10->6->x这条路径上,就会多一个黑色节点,又会破坏规则。那么我们这么想:对于一个黑色节点,如果把该节点变为红色,同时把该节点的两个左右节点变为黑色,那么其实不会改变每一条路径上的黑色节点数量。例如:
我们将10变为红色,将6,15变为黑色。这样既满足了x与6不为连续的红色,也满足了每条路径上的黑色节点数量不变。
以上只是变色中的一种情况,同AVL树一样,如果我们将抽象图画出来,其实本质还是一样,不断往上变色,直到符合红黑树的规则即可。该情况的代码如下:
bool _Insert(const pair<K,V>& value)
{if (_root == nullptr){_root = new Node(value);_root->_col = BLACK;num++;return true;}Node* parent = nullptr;Node* cur = _root;while (cur){if (value < cur->_value){parent = cur;cur = cur->_left;}else if (cur->_value < value){parent = cur;cur = cur->_right;}else{return false;}}cur = new Node(value);if (cur->_value < parent->_value){parent->_left = cur;}else{parent->_right = cur;}cur->_parent = parent; //节点已经被插入,下面考察是否符合红黑树规则while (parent && parent->_col == RED){Node* grandfather = parent->_parent;//找到父节点的父亲if (parent == grandfather->_left)//如果父亲在grandfather的左边{Node* uncle = grandfather->_right;//那么uncle就在grandfather右边if (uncle && uncle->_col == RED)//如果uncle存在并且为红色,一直往上变色即可{parent->_col = uncle->_col = BLACK;grandfather->_col = RED;cur = grandfather;parent = cur->_parent;}//... ...
2.2.3情况2:单旋+变色
该情况下,肯定c和p为两个连续的红色节点,但此时g为黑,u不存在或者u存在且为黑。如果u存在且为黑,那么c一定不是新插入节点,否则就会所有路径黑色节点数量不同,如果u不存在,则c一定是新插入节点。
分析:和情况1一样,这里的p必须变为黑色,才能解决连续红色节点的问题,u不存在或者存在且为黑,这里单纯的变色就无法解决问题,因此我们需要旋转+变色。

对于上方左右两种情况,因为g为黑,因此我们进行单旋之后,g就会变成p的子树,此时我们把g变红,把p变黑即可解决问题。并且我们不需要继续向上更新,因为p已经变成这棵子树的根,且为黑色节点,一定不会和上方产生矛盾。

先看上方的图,如果c为新插入的节点,且u存在为黑,那么c左右都为空,红黑树规则就会被打破。所以c之前为黑,然后从下方子树一直往上变为红。
如果u不存在,那么c一定是新插入的节点。如果c不是新插入的节点,那么就说明插入之前就存在红黑树符合规则,所以c和p不连续为红,此时就不符合所有路径黑色节点数量相等,因为g的右边为空,所以假设不成立,c一定是新插入的节点。
然后我们再看旋转+变色之后的结果,都符合了红黑树的规则。左单旋与上面类似。
代码如下:
if (parent == grandfather->_left)
{Node* uncle = grandfather->_right;if (uncle && uncle->_col == RED)//情况1{parent->_col = uncle->_col = BLACK;grandfather->_col = RED;cur = grandfather;parent = cur->_parent;}else{if (cur == parent->_left)//情况2{RotateR(grandfather);parent->_col = BLACK;grandfather->_col = RED;break;}else//情况3{RotateL(parent);RotateR(grandfather);cur->_col = BLACK;grandfather->_col = RED;break;}}
}
2.2.4情况3:双旋+变色
情况3还是,c,p为红,g为黑,u不存在或者存在且为黑。u不存在,c一定是新增节点,u存在且为黑,c一定不是新增节点。

这里和情况2类似,我们只考虑左边的情况(右边逻辑类似):

我们学过AVL树都知道,尽管不变色,对于这种 < 类型的树,我们必须进行双旋,才能解决问题,< 进行 左右双旋,然后根据结果来看,我们只需要将c变为黑色,将p,g变为红色即可。
之后就是当p为g的右边,逻辑只是和上面的相反,这里就之间展示整个插入的代码:
bool _Insert(const pair<K,V>& value)
{if (_root == nullptr){_root = new Node(value);_root->_col = BLACK;num++;return true;}Node* parent = nullptr;Node* cur = _root;while (cur){if (value < cur->_value){parent = cur;cur = cur->_left;}else if (cur->_value < value){parent = cur;cur = cur->_right;}else{return false;}}cur = new Node(value);if (cur->_value < parent->_value){parent->_left = cur;}else{parent->_right = cur;}cur->_parent = parent;//插入完毕,准备检查是否为规则红黑树while (parent && parent->_col == RED){Node* grandfather = parent->_parent;if (parent == grandfather->_left)//如果p在g的左边{Node* uncle = grandfather->_right;if (uncle && uncle->_col == RED)//u存在且为红,一直向上变色{parent->_col = uncle->_col = BLACK;grandfather->_col = RED;cur = grandfather;parent = cur->_parent;}else//u不存在或者存在且为黑{if (cur == parent->_left)//c为p的左,右单旋+变色{RotateR(grandfather);parent->_col = BLACK;grandfather->_col = RED;break;}else//c为p的右,左右双旋+变色{RotateL(parent);RotateR(grandfather);cur->_col = BLACK;grandfather->_col = RED;break;}}}// g// u p// c//else//p为g的右{Node* uncle = grandfather->_left;if (uncle && uncle->_col == RED)//u存在且为红,一直向上变色{parent->_col = uncle->_col = BLACK;grandfather->_col = RED;cur = grandfather;parent = cur->_parent;}else//u不存在或者存在且为黑{if (cur == parent->_right)//c为p的右,左单旋+变色{RotateL(grandfather);parent->_col = BLACK;grandfather->_col = RED;break;}else//c为p的左,右左双旋+变色{RotateR(parent);RotateL(grandfather);cur->_col = BLACK;grandfather->_col = RED;break;}}}}_root->_col = BLACK;//无论如何,根节点都为黑色num++;//统计红黑树的节点数量return true;
}
2.3红黑树的查找
红黑树的查找和AVL树类似,查找效率为O(),代码如下:
Node* find(const K& key)
{Node* cur = _root;while (cur){if (key < cur->_value.first){cur = cur->_left;}else if (key > cur->_value.first){cur = cur->_right;}else{return cur;}}return nullptr;
}
以上内容若有错误,欢迎批评指正!
相关文章:
红黑树 学习笔记
目录 1.红黑树的概念 1.1红黑树的规则 1.2红黑树的效率 2.红黑树的实现 2.1红黑树的大致结构 2.2红黑树的插入 2.2.1红黑树插入的大致过程 2.2.2情况1:变色 2.2.3情况2:单旋+变色 2.2.4情况3:双旋变色 2.3红黑树的查找…...
linux更改系统时间
测试环境和生产环境代码完全一致,但是生产环境代码碰到了问题,报错类似time expired,猜测和系统时间有关系,修改之后确实好了。测试如下: 参考:centos7时间同步教程_centos7 时间同步,如果遇到…...
B站C#刘铁猛笔记
C#——刘铁猛笔记 类、名称空间(简述) 类(class)是构成程序的主体 名称空间(namespace)以树形结构组织类(其他类型) 名称空间:名称空间是用来组织和管理类、接口、结构…...
如何使用信号发生器产生正弦波并用数字示波器进行测量
使用信号发生器产生正弦波并用数字示波器进行测量的步骤如下: 1. 准备工作 所需设备 信号发生器数字示波器探头(通常为10X衰减探头)BNC电缆和适配器(如果需要) 2. 设置信号发生器 连接 使用BNC电缆将信号发生器的…...
XJ04、消费金融|授信基本概念及其流程设计
银行是经营风险的特殊行业,而银行授信则与银行业务和风险天然相伴。它是银行与客户建立业务关系的起点,也是银行风险管理的关键环节和核心要素。若要了解银行业务,就得先了解银行的授信业务;若要理解银行经营,就得先理…...
儿童预防接种预约微信小程序springboot+论文源码调试讲解
2相关技术 2.1微信小程序 小程序是一种新的开放能力,开发者可以快速地开发一个小程序。小程序可以在微信内被便捷地获取和传播,同时具有出色的使用体验。尤其拥抱微信生态圈,让微信小程序更加的如虎添翼,发展迅猛。 2.2 MYSQL数据…...
nginx 修改配置
如果你的后端服务在不同的端口上运行,但静态资源访问路径相同,你可以使用 Nginx 的 location 配置来将请求转发到不同的后端服务,同时处理静态文件。这里有几种常见的方式: 方案 1: 基于路径的配置 如果所有服务的静态资源路径相…...
孤岛架构在安全性方面
孤岛架构在安全性方面的考虑主要涉及如何确保每个孤岛的安全性,同时维护整个系统的安全。 关键的安全性考虑: 1. 数据隔离和访问控制 数据隔离:每个孤岛应该有独立的数据存储,以确保数据隔离。这有助于防止数据泄露和未经授权的…...
COSCon'24 志愿者招募令:共创开源新生活!
亲爱的开源爱好者们, 第九届中国开源年会(COSCon24)即将在北京中关村国家自主创新示范区会议中心于2024年11月2日至3日隆重举行。今年的主题是“Open Source, Open Life|开源新生活”,旨在探索开源技术如何在各个领域推…...
vscode使用make编译c的问题
问题1:makefile:2: *** missing separator. Stop vscode的配置问题,看这哥们的文章即可:https://blog.csdn.net/m0_57464986/article/details/134220676 问题2:创建makefile文件 直接创建文件名为“makefile”的文件即可&#x…...
管家婆财贸ERP BB019.操作员制单日期控制
最低适用版本: 财贸系列 20.0 插件简要功能说明: 定制操作员权限功能,根据服务器日期控制系统单据新增和修改更多细节描述见下方详细文档 插件操作视频: 进销存类定制插件--操作员制单日期控制 插件详细功能文档: …...
从 Vue 2 到 Vue 3:全面升级指南
🌈个人主页:前端青山 🔥系列专栏:Vue篇 🔖人终将被年少不可得之物困其一生 依旧青山,本期给大家带来Vuet篇专栏内容:Vue-从 Vue 2 到 Vue 3:全面升级指南 前言 随着前端技术的不断发展,Vue.j…...
Apache paimon表操作实战-5
维表Join Paimon支持Lookup Join语法,它用于从 Paimon 查询的数据来补充维度字段。要求一个表具有处理时间属性,而另一个表由查找源连接器支持。 Paimon 支持 Flink 中具有主键的表和append-only的表查找联接。以下示例说明了此功能。 USE CATALOG fs_catalog; CREATE TABL…...
阿里云用STS上传oss的完整程序执行流程图 和前端需要哪些参数uniapp
H5 微信小程序可用的前端直传阿里云OSS(STS临时凭证前端签名)直接下载插件 下面是原理说明: 明白了,我来详细说明前端上传文件到阿里云OSS需要携带的具体参数: 从服务器获取的 STS 凭证: // 这些参数需要从你的后端服务器获…...
决策树方法根据指定条件筛选方案
代码功能说明 条件类:Condition 类用于定义每个条件的范围,并提供一个方法 is_satisfied 来检查输入值是否满足该条件。 算法选择器类:AlgorithmSelector 类负责应用条件并记录不满足的条件。它提供方法 apply_condition 用于更新可用算法&a…...
多特征变量序列预测(四) Transformer-BiLSTM风速预测模型
往期精彩内容: 时序预测:LSTM、ARIMA、Holt-Winters、SARIMA模型的分析与比较 全是干货 | 数据集、学习资料、建模资源分享! EMD、EEMD、FEEMD、CEEMD、CEEMDAN的区别、原理和Python实现(一)EMD-CSDN博客 EMD、EEM…...
【开源免费】基于SpringBoot+Vue.JS蜗牛兼职平台 (JAVA毕业设计)
本文项目编号 T 034 ,文末自助获取源码 \color{red}{T034,文末自助获取源码} T034,文末自助获取源码 目录 一、系统介绍1.1 平台架构1.2 管理后台1.3 用户网页端1.4 技术特点 二、演示录屏三、启动教程四、功能截图五、文案资料5.1 选题背景…...
Ajax笔记
介绍 Ajax是一种网页开发技术,全称是Asynchronous JavaScript and XML(异步JavaScript和XML)。作用如下: 数据交换:可以通过Ajax给服务器发送请求,并获取服务器响应的数据。即前端动态的发送Ajax到服务器端…...
软考:缓存分片和一致性哈希
缓存分片技术是一种将数据分散存储在多个节点上的方法,它在分布式缓存系统中尤为重要。这项技术的核心目的是提高系统的性能和可扩展性,同时确保数据的高可用性。以下是缓存分片技术的一些关键点: 数据分片:缓存分片涉及将数据分成…...
3109 体验积分值
经验值:1200 时间限制:1000毫秒 内存限制:128MB 合肥市第34届信息学竞赛(2017年) 不许抄袭,一旦发现,直接清空经验! 题目描述 Description 卡卡西和小朋友们做完了烧脑的数字游…...
在鸿蒙HarmonyOS 5中实现抖音风格的点赞功能
下面我将详细介绍如何使用HarmonyOS SDK在HarmonyOS 5中实现类似抖音的点赞功能,包括动画效果、数据同步和交互优化。 1. 基础点赞功能实现 1.1 创建数据模型 // VideoModel.ets export class VideoModel {id: string "";title: string ""…...
令牌桶 滑动窗口->限流 分布式信号量->限并发的原理 lua脚本分析介绍
文章目录 前言限流限制并发的实际理解限流令牌桶代码实现结果分析令牌桶lua的模拟实现原理总结: 滑动窗口代码实现结果分析lua脚本原理解析 限并发分布式信号量代码实现结果分析lua脚本实现原理 双注解去实现限流 并发结果分析: 实际业务去理解体会统一注…...
精益数据分析(97/126):邮件营销与用户参与度的关键指标优化指南
精益数据分析(97/126):邮件营销与用户参与度的关键指标优化指南 在数字化营销时代,邮件列表效度、用户参与度和网站性能等指标往往决定着创业公司的增长成败。今天,我们将深入解析邮件打开率、网站可用性、页面参与时…...
面向无人机海岸带生态系统监测的语义分割基准数据集
描述:海岸带生态系统的监测是维护生态平衡和可持续发展的重要任务。语义分割技术在遥感影像中的应用为海岸带生态系统的精准监测提供了有效手段。然而,目前该领域仍面临一个挑战,即缺乏公开的专门面向海岸带生态系统的语义分割基准数据集。受…...
Golang——6、指针和结构体
指针和结构体 1、指针1.1、指针地址和指针类型1.2、指针取值1.3、new和make 2、结构体2.1、type关键字的使用2.2、结构体的定义和初始化2.3、结构体方法和接收者2.4、给任意类型添加方法2.5、结构体的匿名字段2.6、嵌套结构体2.7、嵌套匿名结构体2.8、结构体的继承 3、结构体与…...
【SpringBoot自动化部署】
SpringBoot自动化部署方法 使用Jenkins进行持续集成与部署 Jenkins是最常用的自动化部署工具之一,能够实现代码拉取、构建、测试和部署的全流程自动化。 配置Jenkins任务时,需要添加Git仓库地址和凭证,设置构建触发器(如GitHub…...
通过MicroSip配置自己的freeswitch服务器进行调试记录
之前用docker安装的freeswitch的,启动是正常的, 但用下面的Microsip连接不上 主要原因有可能一下几个 1、通过下面命令可以看 [rootlocalhost default]# docker exec -it freeswitch fs_cli -x "sofia status profile internal"Name …...
如何配置一个sql server使得其它用户可以通过excel odbc获取数据
要让其他用户通过 Excel 使用 ODBC 连接到 SQL Server 获取数据,你需要完成以下配置步骤: ✅ 一、在 SQL Server 端配置(服务器设置) 1. 启用 TCP/IP 协议 打开 “SQL Server 配置管理器”。导航到:SQL Server 网络配…...
2025年- H71-Lc179--39.组合总和(回溯,组合)--Java版
1.题目描述 2.思路 当前的元素可以重复使用。 (1)确定回溯算法函数的参数和返回值(一般是void类型) (2)因为是用递归实现的,所以我们要确定终止条件 (3)单层搜索逻辑 二…...
宠物车载安全座椅市场报告:解读行业趋势与投资前景
一、什么是宠物车载安全座椅? 宠物车载安全座椅是一种专为宠物设计的车内固定装置,旨在保障宠物在乘车过程中的安全性与舒适性。它通常由高强度材料制成,具备良好的缓冲性能,并可通过安全带或ISOFIX接口固定于车内。 近年来&…...
