0

0

c++如何实现红黑树算法_c++ 数据结构定义与节点旋转逻辑【详解】

冰火之心

冰火之心

发布时间:2026-01-02 17:24:02

|

904人浏览过

|

来源于php中文网

原创

红黑树节点必须包含颜色字段和父指针;旋转需更新四元组父子关系;插入修复分四类case处理;删除复杂,建议优先使用std::map/set。

c++如何实现红黑树算法_c++ 数据结构定义与节点旋转逻辑【详解】

红黑树节点定义必须包含颜色字段和三叉指针

标准二叉搜索树节点不满足红黑树约束,RBNode 必须显式存储颜色(通常用 bool is_red 或枚举),且需保留父指针——这是实现自底向上修复旋转的前提。子节点指针命名建议统一为 leftright,避免用 lchild 等非主流写法导致后续逻辑混乱。

常见错误:漏掉 parent 指针,导致插入后无法回溯调整;或把颜色存在 int color 里却未定义 RED=0/BLACK=1,引发条件判断歧义。

struct RBNode {
    int key;
    bool is_red;  // true 表示红色,false 表示黑色
    RBNode* left;
    RBNode* right;
    RBNode* parent;
RBNode(int k) : key(k), is_red(true), left(nullptr), right(nullptr), parent(nullptr) {}

};

左旋与右旋函数必须更新父子关系四元组

旋转不是单纯交换指针,而是重连 old_rootnew_rootsubtreeparent 四者之间的 left/right/parent 字段。漏掉任一链接(尤其是 new_root->parentold_root->parent)都会导致树断裂或内存泄漏。

立即学习C++免费学习笔记(深入)”;

  • left_rotate(x):以 x 为轴,将 x->right 提升为新根,x 成为其左子
  • right_rotate(y):以 y 为轴,将 y->left 提升为新根,y 成为其右子
  • x 是根节点,旋转后需同步更新树的 root 指针
void left_rotate(RBNode*& root, RBNode* x) {
    RBNode* y = x->right;
    x->right = y->left;
    if (y->left != nullptr) y->left->parent = x;
y->parent = x->parent;
if (x->parent == nullptr) {
    root = y;
} else if (x == x->parent->left) {
    x->parent->left = y;
} else {
    x->parent->right = y;
}

y->left = x;
x->parent = y;

}

魔术橡皮擦
魔术橡皮擦

智能擦除、填补背景内容

下载

插入后修复需分四类 case 处理,核心是变色+旋转组合

插入新节点默认设为红色,仅可能违反「红节点不能有红子」和「根必须为黑」两条规则。修复函数 insert_fixup 从新节点开始向上迭代,关键在于识别当前节点 z、父节点 z->parent、叔节点 uncle 的颜色组合:

  • Case 1:叔节点为红 → 父与叔变黑,祖父变红,z = z->parent->parent 继续向上检查
  • Case 2:叔为黑且 z 是右子 → 先 left_rotate(z->parent),转为 Case 3
  • Case 3:叔为黑且 z 是左子 → right_rotate(z->parent->parent),再将父变黑、祖父变红
  • 注意:所有旋转后必须立即修正颜色,且 Case 1 中若祖父为根,则最后一步需强制置黑

容易被忽略的点:uncle 判空必须用 uncle == nullptr 而非 !uncle(避免未初始化野指针误判);修复循环终止条件是 z->parent == nullptr || !z->parent->is_red

删除操作比插入复杂得多,建议优先复用 std::map/set

红黑树删除涉及「找后继→替换→修复双黑」三阶段,修复逻辑包含至少 6 种 case,其中「双黑传播」需同时处理兄弟颜色、兄弟子节点颜色、兄弟是否为根等交叉条件。手动实现极易在边界场景(如删根、删唯一子树、连续黑节点)出错。

除非教学目的或嵌入式环境严格限制 STL,否则直接使用 std::mapstd::set 更可靠——它们底层就是红黑树,接口稳定,且经过数十年编译器厂商验证。自己造轮子时,务必用大量随机序列 + valgrind 检查内存泄漏和非法访问。

相关文章

c++速学教程(入门到精通)
c++速学教程(入门到精通)

c++怎么学习?c++怎么入门?c++在哪学?c++怎么学才快?不用担心,这里为大家提供了c++速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

312

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

522

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

49

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

190

2025.08.29

treenode的用法
treenode的用法

​在计算机编程领域,TreeNode是一种常见的数据结构,通常用于构建树形结构。在不同的编程语言中,TreeNode可能有不同的实现方式和用法,通常用于表示树的节点信息。更多关于treenode相关问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

529

2023.12.01

C++ 高效算法与数据结构
C++ 高效算法与数据结构

本专题讲解 C++ 中常用算法与数据结构的实现与优化,涵盖排序算法(快速排序、归并排序)、查找算法、图算法、动态规划、贪心算法等,并结合实际案例分析如何选择最优算法来提高程序效率。通过深入理解数据结构(链表、树、堆、哈希表等),帮助开发者提升 在复杂应用中的算法设计与性能优化能力。

12

2025.12.22

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

994

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

53

2025.10.17

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

74

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
HTML5/CSS3/JavaScript/ES6入门课程
HTML5/CSS3/JavaScript/ES6入门课程

共102课时 | 6.6万人学习

前端基础到实战(HTML5+CSS3+ES6+NPM)
前端基础到实战(HTML5+CSS3+ES6+NPM)

共162课时 | 18.5万人学习

第二十二期_前端开发
第二十二期_前端开发

共119课时 | 12.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号