0

0

C++怎么实现红黑树_C++自平衡树教程【结构】

穿越時空

穿越時空

发布时间:2026-03-06 11:02:03

|

702人浏览过

|

来源于php中文网

原创

写红黑树关键在于严格维护五条性质,每次插入/删除后必须显式恢复;漏一条会导致erase死循环或树高异常,核心是旋转更新parent指针、修复前判空、颜色操作精准、异常安全保证。

c++怎么实现红黑树_c++自平衡树教程【结构】

红黑树不是靠“教程结构”写出来的,是靠五个不变量硬扛出来的

写一个能用的红黑树,关键不在画结构图或背插入流程,而在每次插入/删除后,**必须显式恢复那五条红黑性质**。漏一条,find可能不崩,但erase大概率死循环,或者insert后树高突然跳变——这不是bug,是性质坍塌。

实操建议:

九歌
九歌

九歌--人工智能诗歌写作系统

下载
  • 先写好Node结构:至少含colorbool或枚举)、leftrightparent;空节点用nullptr,别搞哨兵节点——初学者加哨兵反而掩盖指针错误
  • 把五条性质写成注释贴在代码顶部:root is blackred node's children are blackall paths from node to leaves have same black count等,每次改完立刻对照检查
  • 旋转函数(rotate_leftrotate_right)必须返回新子树根,且要同步更新parent指针;常见错误是只改了left/right,忘了new_root->parent = old_parent

插入后修复:90%的崩溃来自“忘掉祖父节点的颜色判断”

标准插入后修复分三种情况(LL/LR/RR/RL归为两类),但真正卡住人的,是进入修复前没确认grandparent是否存在,以及它是不是红色——如果grandparent == nullptr或它是黑色,直接return,不该进任何case。

常见错误现象:

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

  • Segmentation faultif (grandparent->color == RED)这行——因为grandparentnullptr
  • 插入后树看起来平衡,但再插一个就double free——本质是某次旋转中parent指针没重置,导致两个节点同时指向同一子树

实操建议:

  • 修复函数开头强制检查:if (!grandparent || grandparent->color == BLACK) return;
  • 所有涉及uncle的分支,先判uncle != nullptr,再取uncle->color;别信“uncle一定存在”
  • 颜色翻转(recolor)时,只动parentunclegrandparent三者;别顺手把current也改了——它此时是红色,改完就违反“红节点孩子必黑”

删除后修复:比插入多一层抽象,核心是“黑高补偿”

删除难点不在找后继,而在被删节点deleted的替代节点replacement若为黑色,会导致以它为根的子树“黑高-1”。修复目标不是把颜色填回去,而是通过旋转+ recolor,让整棵树各路径黑节点数重新一致。

使用场景:

  • 调用erase(key)后程序卡住或返回错误迭代器——大概率fix_double_black没处理replacement == nullptr的情况(即删的是叶子)
  • 删完再查,有些key搜不到——说明某次旋转后parent没更新,导致子树脱离主干

实操建议:

  • 删除函数末尾统一调用fix_double_black(replacement),哪怕replacementnullptr;该函数第一行就得处理if (!node) { node = parent; },否则空指针解引用
  • 修复中的四种case(node是双重黑,sibling颜色+子节点颜色组合)必须覆盖全;漏掉sibling为黑但两个孩子都黑的情况,就会无限循环
  • 旋转后务必重置node的父指针,例如rotate_left(parent)后,node->parent仍是原parent,但parent已不是它的父了——得手动设node->parent = new_parent

C++实现里最容易被忽略的:移动语义和异常安全

标准库std::map能安全抛异常,是因为它所有操作都满足强异常保证;你自己写的红黑树如果支持T含非noexcept构造/赋值,insert中途抛异常,就可能留下半残树——比如新节点已链入,但颜色没设,或旋转一半就停了。

性能与兼容性影响:

  • std::movevalueinsert,但别在节点构造里直接std::move——万一T移动构造抛异常,节点内存已分配却未初始化
  • 所有修改指针的操作(旋转、重连)必须放在try块外;真正可能抛异常的只有new Node(std::move(value))这一句,其余都是纯指针操作,不会抛
  • 别为了省事把Node做成std::unique_ptr——它会干扰parent指针维护,且增加间接访问开销;裸指针+手动delete更可控

复杂点从来不在旋转逻辑本身,而在于你是否记得:每改一个->left,就得同步改对应子节点的->parent;每动一次颜色,就得确认没破坏任意一条性质;每次抛异常,树必须回到插入前状态。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

841

2023.08.22

counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

203

2023.11.20

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

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

294

2025.08.29

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

105

2025.10.23

空指针异常处理
空指针异常处理

本专题整合了空指针异常解决方法,阅读专题下面的文章了解更多详细内容。

23

2025.11.16

golang map内存释放
golang map内存释放

本专题整合了golang map内存相关教程,阅读专题下面的文章了解更多相关内容。

77

2025.09.05

golang map相关教程
golang map相关教程

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

39

2025.11.16

golang map原理
golang map原理

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

67

2025.11.17

Rust内存安全机制与所有权模型深度实践
Rust内存安全机制与所有权模型深度实践

本专题围绕 Rust 语言核心特性展开,深入讲解所有权机制、借用规则、生命周期管理以及智能指针等关键概念。通过系统级开发案例,分析内存安全保障原理与零成本抽象优势,并结合并发场景讲解 Send 与 Sync 特性实现机制。帮助开发者真正理解 Rust 的设计哲学,掌握在高性能与安全性并重场景中的工程实践能力。

19

2026.03.05

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 10.7万人学习

C 教程
C 教程

共75课时 | 5.2万人学习

C++教程
C++教程

共115课时 | 20.6万人学习

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

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