当前位置: 首页 > news >正文

数据结构:红黑树的插入实现(C++)

在这里插入图片描述

个人主页 : 个人主页
个人专栏 : 《数据结构》 《C语言》《C++》《Linux》

文章目录

  • 一、红黑树
  • 二、红黑树的插入
  • 三、代码实现
  • 总结


一、红黑树

红黑树的概念:
红黑树是一颗二叉搜索树,但在每个节点上增加一个存储位表示节点的颜色,该节点颜色不是红色就是黑色。通过对每一条从根节点到叶子结点路径上各个节点颜色的控制,确保没有一条路径会比其它路径长出两倍,因此红黑树是接近平衡。

在这里插入图片描述

那红黑树是通过哪些规则来对节点颜色进行控制?

  1. 每个节点不是红色就是黑色

  2. 根节点是黑色

  3. 如果一个节点是红色,则其两个子节点是黑色(不能有连续的红色节点)

  4. 对于每个节点,从该节点到其所以叶子结点的路径上,其黑色节点的数目相同

  5. 每个叶子结点都是黑色的(此处的叶子结点是空节点(NIL),方便我们计算路径的个数)
    注意:上述中的路径是从某一节点到NIL节点。如上图8节点到叶子结点就有5条路径,每条路径都有一个黑色节点。

那为什么遵循这5条规则,红黑树就能保证其最长路径中节点的个数不会超过最短路径节点个数的两倍?
我们假设从根节点到叶子结点的黑色节点数为n,那么最短路径不就是连续的黑色节点,最短路径的节点数为n,那么最长路径不就是红黑相间,最长路径的节点数为2n。所以红黑树保证其最长路径中节点的个数不会超过最短路径节点个数的两倍。


下面是红黑树节点的定义。

enum
{RED,BLACK
};template<class T>
struct RBTreeNode
{RBTreeNode(T data = T()):_left(nullptr),_right(nullptr),_parent(nullptr),_data(data),_col(RED){}RBTreeNode* _left;RBTreeNode* _right;RBTreeNode* _parent;T _data;int _col;
};

该定义中,我们默认将新节点颜色定义为红色,这样我们插入节点时需要维护规则的成本就少。如我们新插入一个红色节点,那么有可能会违背规则3(当其父节点是红色时,有连续红色节点),这时我们可能需要一些变色和旋转,来维持规则,但如果我们插入节点是黑色,那么我们一定违背4(每条路径上黑色节点数相同),这时我们可能需要对整个数进行操作。

二、红黑树的插入

红黑树也是一个二叉搜索树,插入新节点与二叉搜索树的操作一样,如果新插入节点比该节点大,新插入节点就去该节点的右子树,反之去该节点的左子树。

if (_root == nullptr){_root = new Node(data);_root->_col = BLACK;return true;}Node* parent = nullptr;Node* cur = _root;while (cur != nullptr){parent = cur;if (cur->_data > data){cur = cur->_left;}else if(cur->_data < data){cur = cur->_right;}else // cur->_data == data{return false;}}cur = new Node(data);cur->_parent = parent;if (cur->_data > parent->_data){parent->_right = cur;}else{parent->_left = cur;}

这里我们主要分析新插入节点后,红黑树的情况和处理。对于旋转,这里并不会详细解析。对于旋转的详细解析在我的AVL树一文中。数据结构:AVLTree的插入和删除的实现

在分析情况前,我们先了解几个节点的含义,方便后续的理解。
在这里插入图片描述
接下来的所有情况都与这四个节点有关。

因为我们新插入的节点是红色,如果新插入节点的父节点是黑色,那么红黑树的规则未被破坏。如果新插入节点的父节点是红色,那么就有连续的红色节点。这是我们就要分情况讨论。

情况1. cur为红色,parent为红色,grandfather为黑色,uncle为红色
也就是如下图所示:
在这里插入图片描述
这种情况是最简单的情况,我们只需要将parent 与 uncle的颜色变成黑色(解决连续红色节点),再将grandfather的颜色变成红色(防止经过grandfather路径的黑色节点数+1)
在这里插入图片描述
但就如上图所示,因为我们将grandfather的颜色变成红色,如果grandfather的父节点的颜色也是红色,这时我们依旧有连续的红色节点,我们仍需对grandfather进行处理。
在这里插入图片描述
我们重复变色过程
在这里插入图片描述
这时,grandfather没有父节点,就可以停止了,但此时grandfather作为根节点为红色,我们需要特殊处理一下即可。这样对于该插入新节点的情况一就完成了。
下面是总结的模型:
在这里插入图片描述
对于这种cur,parent,uncle为红色,grandfather为黑色的情况,我们只需让parent,uncle变成红色,grandfather变成黑色,接着需要检查grandfather的父节点是否是红色,如果是红色,重复上述操作。如果是黑色,就可以结束了。

情况2:cur为红色,parent为红色,grandfather为黑色,uncle不存在或uncle存在且为黑色
在这里插入图片描述
这时情况1 的处理就行不通了,因为uncle要么不存在,要么本身就为黑色,如果将grandfather变成红色,那么经过grandfather的路径的黑色节点数就-1。所以我们要采取旋转+变色。
在这里插入图片描述
因为cur在parent的右侧,parent在grandfather的右侧,成直线。所以我们对grandfather左单旋,接着将parent的颜色变成黑色,grandfather的颜色变成红色(防止经过grandfather的路径的黑色节点数+1),又因为parent作为新的根节点为黑色,所以我们不需要去检查parent的父节点的颜色。(虽然我们也可以只将cur变为黑色,但这样我们就需要检查parent父节点的颜色)
那如果我们新增5节点要怎么处理?
在这里插入图片描述
此时我们也需要旋转+变色,但我们要双旋。
在这里插入图片描述
如上图,我们要先对parent左单旋,使grandfather,cur,parent在同一侧,接着使grandfather左单旋,cur变为黑色,grandfather变成红色。
如果parent在grandfather的左侧,情况与上述情况类似,只需要改变旋转方向即可。
下面是总结的模型:
单旋在这里插入图片描述
双旋
在这里插入图片描述

while (parent != nullptr && parent->_col != BLACK){Node* grandfather = parent->_parent;if (parent == grandfather->_left){Node* uncle = grandfather->_right;if (uncle != nullptr && uncle->_col == RED) // uncle存在且为红色{parent->_col = uncle->_col = BLACK;grandfather->_col = RED;cur = grandfather;parent = cur->_parent;}else if (uncle == nullptr || uncle->_col == BLACK) // uncle不存在 或 umcle存在且为黑{if (cur == parent->_left) // 同方向{RotateR(grandfather);parent->_col = BLACK;grandfather->_col = RED;}else // cur == parent->_right 不同方向{RotateL(parent);RotateR(grandfather);cur->_col = BLACK;grandfather->_col = RED;}break;}}else // parent == grandfather->_right{Node* uncle = grandfather->_left;if (uncle != nullptr && uncle->_col == RED){parent->_col = uncle->_col = BLACK;grandfather->_col = RED;cur = grandfather;parent = cur->_parent;}else if (uncle == nullptr || uncle->_col == BLACK){if (cur == parent->_right){RotateL(grandfather);parent->_col = BLACK;grandfather->_col = RED;}else{RotateR(parent);RotateL(grandfather);cur->_col = BLACK;grandfather->_col = RED;}break;}}}_root->_col = BLACK; //特殊处理,保证根节点是黑色

则此,红黑树的插入就完成了。

三、代码实现

RBTree.h 文件是红黑树插入的实现
test.cpp 文件是测试用例

// RBTree.h 文件
#pragma onceenum
{RED,BLACK
};template<class T>
struct RBTreeNode
{RBTreeNode(T data = T()):_left(nullptr),_right(nullptr),_parent(nullptr),_data(data),_col(RED){}RBTreeNode* _left;RBTreeNode* _right;RBTreeNode* _parent;T _data;int _col;
};template<class T>
class RBTree
{typedef RBTreeNode<T> Node;
public:RBTree():_root(nullptr){}bool insert(const T& data){if (_root == nullptr){_root = new Node(data);_root->_col = BLACK;return true;}Node* parent = nullptr;Node* cur = _root;while (cur != nullptr){parent = cur;if (cur->_data > data){cur = cur->_left;}else if(cur->_data < data){cur = cur->_right;}else // cur->_data == data{return false;}}cur = new Node(data);cur->_parent = parent;if (cur->_data > parent->_data){parent->_right = cur;}else{parent->_left = cur;}while (parent != nullptr && parent->_col != BLACK){Node* grandfather = parent->_parent;if (parent == grandfather->_left){Node* uncle = grandfather->_right;if (uncle != nullptr && uncle->_col == RED) // uncle存在且为红色{parent->_col = uncle->_col = BLACK;grandfather->_col = RED;cur = grandfather;parent = cur->_parent;}else if (uncle == nullptr || uncle->_col == BLACK) // uncle不存在 或 umcle存在且为黑{if (cur == parent->_left) // 同方向{RotateR(grandfather);parent->_col = BLACK;grandfather->_col = RED;}else // cur == parent->_right 不同方向{RotateL(parent);RotateR(grandfather);cur->_col = BLACK;grandfather->_col = RED;}break;}}else // parent == grandfather->_right{Node* uncle = grandfather->_left;if (uncle != nullptr && uncle->_col == RED){parent->_col = uncle->_col = BLACK;grandfather->_col = RED;cur = grandfather;parent = cur->_parent;}else if (uncle == nullptr || uncle->_col == BLACK){if (cur == parent->_right){RotateL(grandfather);parent->_col = BLACK;grandfather->_col = RED;}else{RotateR(parent);RotateL(grandfather);cur->_col = BLACK;grandfather->_col = RED;}break;}}}_root->_col = BLACK;return true;}// 读取红黑树最左侧节点Node* leftMost(){if (_root == nullptr){return nullptr;}Node* cur = _root;while (cur->_left != nullptr){cur = cur->_left;}return cur;}// 读取红黑树最右侧节点Node* rightMost(){if (_root == nullptr){return nullptr;}Node* cur = _root;while (cur->_right != nullptr){cur = cur->_right;}return cur;}bool isValidRBTree(){if (_root->_col == RED){return false;}// 找最左边的黑色节点数作为标准比较int pathBlack = 0;Node* cur = _root;while (cur != nullptr){if (cur->_col == BLACK){pathBlack++;}cur = cur->_left;}return _isValidRBTree(_root, 0, pathBlack);}bool _isValidRBTree(Node* root, int blackCount, int pathBlack){if (root == nullptr){if (blackCount == pathBlack)return true;elsereturn false;}if (root->_col == RED&& root->_parent != nullptr&& root->_parent->_col == RED){cout << "有连续的红色节点" << endl;return false;}if (root->_col == BLACK){blackCount++;}return _isValidRBTree(root->_left, blackCount, pathBlack)&& _isValidRBTree(root->_right, blackCount, pathBlack);}void RotateR(Node* parent){Node* subL = parent->_left;Node* subLR = subL->_right;Node* pparent = parent->_parent;parent->_left = subLR;subL->_right = parent;parent->_parent = subL;if (subLR != nullptr){subLR->_parent = parent;}if (pparent == nullptr){_root = subL;subL->_parent = nullptr;}else{if (pparent->_left == parent){pparent->_left = subL;}else{pparent->_right = subL;}subL->_parent = pparent;}}void RotateL(Node* parent){Node* subR = parent->_right;Node* subRL = subR->_left;Node* pparent = parent->_parent;parent->_right = subRL;subR->_left = parent;parent->_parent = subR;if (subRL != nullptr){subRL->_parent = parent;}if (pparent == nullptr){_root = subR;subR->_parent = nullptr;}else{if (pparent->_left == parent){pparent->_left = subR;}else{pparent->_right = subR;}subR->_parent = pparent;}}private:Node* _root;
};

//test.cpp 文件
#include <iostream>
#include <vector>using namespace std;
#include "RBTree.h"#define N 10000000void test1()
{vector<int> nums(N);srand((unsigned int)time(0));for (int i = 0; i < N; i++){nums[i] = rand() + i;//cout << nums[i] << endl;}RBTree<int> tree;for (auto val : nums){if (val == 11478){int i = 0;i++;}tree.insert(val);//cout << val << "->" << tree.isValidRBTree() << endl;}cout << tree.isValidRBTree() << endl;
}int main()
{test1();return 0;
}

总结

以上就是我对于红黑树插入实现的总结。感谢支持!!!
在这里插入图片描述

相关文章:

数据结构:红黑树的插入实现(C++)

个人主页 &#xff1a; 个人主页 个人专栏 &#xff1a; 《数据结构》 《C语言》《C》《Linux》 文章目录 一、红黑树二、红黑树的插入三、代码实现总结 一、红黑树 红黑树的概念&#xff1a; 红黑树是一颗二叉搜索树&#xff0c;但在每个节点上增加一个存储位表示节点的颜色&…...

飞天使-django之数据库简介

文章目录 增删改查解决数据库不能存储中文问题创建表数据类型表的基本操作主键唯一键 unique外键实战 增删改查 四个常用的语句查询 : insert delete update select insert into student(Sno,name) values(95001,"张三") delete from student where name张三 upda…...

Flink之KeyedState

前面的文章中介绍过Operator State,这里介绍一下Keyed State. 在使用Operator State时必须要实现CheckpointFunction接口,而Keyed State则不需要,在使用keyBy(...)分组分组后,调用的函数必须是实现RichFuntion接口的函数才可以使用Keyed State.同样使用Keyed State也必须开启Ch…...

c语言:模拟实现qsort函数

qsort函数的功能&#xff1a; qsort相较于冒泡排序法&#xff0c;不仅效率更快&#xff0c;而且能够比较不同类型的元素&#xff0c;如&#xff1a;浮点数&#xff0c;结构体等等。这里我们来模拟下qsort是如何实现这一功能的&#xff0c;方便我们对指针数组有一个更深层次的理…...

从0开始学习数据结构 C语言实现 1.前篇及二分查找算法

一、前篇 1、什么是数据结构&#xff1f; 数据结构是带有结构特性的数据元素的集合&#xff0c;它研究的是数据的逻辑结构和数据的物理结构以及它们之间的相互关系 2、时间复杂度与空间复杂度 大O符号是用于描述函数渐进行为的数学符号 常用函数的增长表 阶乘O(n!) > 指数…...

VSCode 使用CMakePreset找不到cl.exe编译器的问题

在用vscode开发c项目的时候&#xff0c;使用预先配置的CMakePresets.json可以把一些特定的cmake选项固定下来&#xff0c;在配置时直接使用 "cmake --config --preset presetname"就可以进行配置&#xff0c;免去在命令行输入过多的配置参数。 但是在vscode中&#…...

【Linux系统化学习】进程的状态 | 僵尸进程 | 孤儿进程

个人主页点击直达&#xff1a;小白不是程序媛 Linux专栏&#xff1a;Linux系统化学习 目录 操作系统进程的状态 运行状态 阻塞状态 进程阻塞的现象 挂起阻塞状态 Linux进程状态 Linux内核源代码怎么说 R&#xff08;running状态&#xff09;运行状态 S&#xff08;sl…...

深信服AC流量管理技术

拓扑图 一.保证通道针对修仙部&#xff0c;访问网站&#xff0c;邮件&#xff0c;DNS&#xff0c;IM&#xff0c;办工 OA&#xff0c;微博论坛网上银行等常见应用保证带宽最低 50%&#xff0c;最高 100% 1. 先新建线路带宽 2.新增流量管理通道&#xff08;保证关键应用&#x…...

二元关系及关系代数中的象集、除运算

二元关系及关系代数中的象集、除运算 数学上&#xff0c;二元关系用于讨论两个数学对象的联系。诸如算术中的「大于」及「等于」&#xff0c;几何学中的"相似"&#xff0c;或集合论中的"为...之元素"或"为...之子集"。二元关系有时会简称关系&a…...

[PHP]关联和操作MySQL数据库然后将数据库部署到ECS

在Mac电脑上使用VS Code进行PHP开发并关联操作MySQL数据库&#xff0c;然后将数据库部署到ECS。 1.安装PHP和MySQL 确保你的Mac上已经安装了PHP和MySQL。你可以使用Homebrew来安装它们&#xff1a; $ brew install php $ brew install mysql 安装mysql完成后记住这一句: …...

23.11.19日总结

经过昨天的中期答辩&#xff0c;其实可以看出来项目进度太慢了&#xff0c;现在是第十周&#xff0c;预计第十四周是终级答辩&#xff0c;在这段时间要把项目写完。 前端要加上一个未登录的拦截器&#xff0c;后端加上全局的异常处理。对于饿了么项目的商品建表&#xff0c;之前…...

系列一、JVM概述

一、概述 1.1、Java发展中的重大事件 1.2、虚拟机 vs Java虚拟机 1.2.1、虚拟机 1.2.2、Java虚拟机 1.2.3、Java虚拟机的作用 Java虚拟机是二进制字节码的运行环境&#xff0c;负责装载字节码到其内部&#xff0c;解释/编译为对应平台上的机器指令指令。每一条Java指令&#…...

milvus数据管理-压缩数据

Milvus 默认支持自动数据压缩。您可以 配置 Milvus 以启用或禁用 压缩 和自动压缩。 如果自动压缩被禁用&#xff0c;您仍然可以手动压缩数据。 1.手动压缩数据 压缩请求是异步处理的&#xff0c;因为它们通常需要花费很长时间。 from pymilvus import Collection collection…...

SpringBoot项目连接linux服务器数据库两种解决方法(linux直接开放端口访问本机通过SSH协议访问,以mysql为例)

最近找个springboot脚手架重新熟悉一下springboot相关框架的东西&#xff0c;结果发现好像项目还不能直接像数据库GUI工具一样填几个SSH参数就可以了&#xff0c;于是就给他再整一下看看如何解决 linux开放3306&#xff08;可修改&#xff09;端口直接访问 此方法较为方便&am…...

【Rust】快速教程——闭包与生命周期

前言 你怎么向天生的瞎子说清颜色&#xff1f;怎么用手势向天生的聋子描述声音&#xff1f; 鲜花就在眼前&#xff0c;雷鸣就在头顶&#xff0c;对他们来说却都毫无意义 眼睛看不到&#xff0c;鼻子可以嗅闻花香&#xff0c;耳朵听不见&#xff0c;手指可以触碰窗纸的震动。 犯…...

redis高级案列case

案列一 双写一致性 案例二 双锁策略 package com.redis.redis01.service;import com.redis.redis01.bean.RedisBs; import com.redis.redis01.mapper.RedisBsMapper; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; imp…...

Vue3+Vite实现工程化,attribute属性渲染v-bind指令

想要渲染一个元素的attribute&#xff0c;应该使用v-bind指令 由于插值表达式不能直接放在标签的属性中&#xff0c;所有要渲染元素的属性就应该使用v-bindv-bind可以用于渲染任何元素的属性&#xff0c;语法为 v-bind:属性名数据名&#xff0c;可以简写为 :属性名数据名 <…...

下一代搜索引擎会什么?

现在是北京时间2023年11月18日。聊一聊搜索。 说到搜索&#xff0c;大家首先想到的肯定是谷歌&#xff0c;百度。我把这些定义成上一个时代的搜索引擎。ChatGPT已经火热了有一年的时间了&#xff0c;大家都认为Ai搜索是下一代的搜索。但是AI搜索&#xff0c;需要的是很大算力&a…...

WPF中如何在MVVM模式下关闭窗口

完全来源于十月的寒流&#xff0c;感谢大佬讲解 使用Behaviors <Window x:Class"Test_03.MainWindow"xmlns"http://schemas.microsoft.com/winfx/2006/xaml/presentation"xmlns:b"http://schemas.microsoft.com/xaml/behaviors"xmlns:x&quo…...

【数据结构&C++】二叉平衡搜索树-AVL树(25)

前言 大家好吖&#xff0c;欢迎来到 YY 滴C系列 &#xff0c;热烈欢迎&#xff01; 本章主要内容面向接触过C的老铁 主要内容含&#xff1a; 欢迎订阅 YY滴C专栏&#xff01;更多干货持续更新&#xff01;以下是传送门&#xff01; 目录 一.AVL树的概念二.AVL树节点的定义(代码…...

Win11升级还是全新安装?保姆级决策指南与数据迁移全流程

Win11升级还是全新安装&#xff1f;保姆级决策指南与数据迁移全流程 每次Windows重大版本更新&#xff0c;用户都会面临一个经典难题&#xff1a;是选择保留数据的平滑升级&#xff0c;还是彻底格式化重装系统&#xff1f;这个问题在Win11时代尤为突出——新系统带来的界面革新…...

OpenAI收购科技脱口秀TBPN,力图塑造AI叙事话语权

OpenAI正通过收购备受硅谷内部人士关注的科技脱口秀TBPN进军媒体行业&#xff0c;该节目主持人周三宣布了这一消息。联合主持人约翰库根和乔迪海斯每个工作日从洛杉矶直播TBPN节目三小时&#xff0c;邀请的嘉宾包括创业者、风险投资家和科技界重要人物。此次交易的财务条款未予…...

[Android] 故宫陶瓷馆 v2.2.251126

[Android] 故宫陶瓷馆 v2.2.251126 链接&#xff1a;https://pan.xunlei.com/s/VOpHzrBozQgvaUJbdCkB20SMA1?pwdu338# 故宫陶瓷馆是故宫博物院官方出品的APP&#xff0c;以“时间轴”为核心骨架、全新技术手段打造的陶瓷馆&#xff0c;为你将展品带至手中、带至眼前。...

表格设计:结构与美感并重

1. 表格的结构如果把表格比作一座建筑&#xff0c;那么它的每个结构部件都承担着特定功能。下面是一个完整的表格示例&#xff0c;展示了所有标准结构组件&#xff1a;表格结构图解&#xff1a;标题与副标题&#xff1a;表格的"名字"和"简介"&#xff0c;告…...

Arduino轻量级CRC-32校验库:零依赖、低内存、确定性执行

1. 项目概述Arduino_CRC32 是一个面向嵌入式场景轻量级 CRC-32 校验库&#xff0c;专为 Arduino 及兼容平台&#xff08;如 STM32 Core for Arduino、ESP32 Arduino Core&#xff09;设计。其核心目标并非追求极致吞吐性能&#xff0c;而是以零依赖、低内存占用、确定性执行时间…...

OpenClaw × 88API:10 分钟搭好本地网关,解决 API 超时和多渠道切换(2026 完整教程)

你可能也踩过这些坑&#xff1a;项目快提测了&#xff0c;Claude API 突然超时&#xff0c;重试半天还是报错想临时换一个中转站兜底&#xff0c;结果又要改一遍 base_url、api_key、模型名一个渠道支持 Claude&#xff0c;不支持 Gemini&#xff1b;另一个支持 GPT&#xff0c…...

3个颠覆级提速方案:ComfyUI-Manager下载性能优化指南

3个颠覆级提速方案&#xff1a;ComfyUI-Manager下载性能优化指南 【免费下载链接】ComfyUI-Manager ComfyUI-Manager is an extension designed to enhance the usability of ComfyUI. It offers management functions to install, remove, disable, and enable various custom…...

LoRa土壤监测与灌溉控制系统方案

当前农业生产中&#xff0c;土壤水分、温度等环境参数是影响作物生长的核心因素&#xff0c;传统种植模式依赖人工经验判断灌溉时机与用量&#xff0c;存在诸多局限。随着智慧农业、精准农业的快速发展&#xff0c;物联网技术在农业灌溉领域的应用日益广泛&#xff0c;LoRa作为…...

番茄小说下载器:开源电子书工具全解析

番茄小说下载器&#xff1a;开源电子书工具全解析 【免费下载链接】Tomato-Novel-Downloader 番茄小说下载器不精简版 项目地址: https://gitcode.com/gh_mirrors/to/Tomato-Novel-Downloader 番茄小说下载器是一款基于Rust语言开发的开源工具&#xff0c;专为解决在线小…...

汽车行业空气动力学仿真Fluent的license分点方案

汽车行业空气动力学仿真Fluent的License分点方案你是绝非老是在项目高峰时段发现Fluent的License不够用了&#xff0c;而且平时又有数来空闲许可在浪费&#xff1f;你是不光是也在担心合规风险&#xff0c;搞不好一不小心就超了额度&#xff0c;被软件商追着要钱&#xff1f;实…...