0

0

c++如何实现一个红黑树 c++关联容器底层【数据结构】

裘德小鎮的故事

裘德小鎮的故事

发布时间:2025-12-27 17:03:07

|

624人浏览过

|

来源于php中文网

原创

c++kquote>红黑树是C++标准库中std::map、std::set等关联容器的底层实现,是一种通过红黑着色与旋转变色维持平衡的二叉搜索树,满足五条不变性,插入删除查找均为O(log n),支持有序遍历与稳定迭代器。

c++如何实现一个红黑树 c++关联容器底层【数据结构】

红黑树在 C++ 标准库中是 std::mapstd::set 等关联容器的底层实现(GCC libstdc++ 和 Clang libc++ 均采用红黑树),它是一种自平衡二叉搜索树,通过节点着色(红/黑)和局部旋转+变色操作,在 O(log n) 时间内保证插入、删除、查找的最坏性能。

红黑树的核心性质

理解实现前必须牢记五条不变性(即“红黑性质”):

  • 每个节点非红即黑
  • 根节点是黑色
  • 所有叶子(NIL 节点,通常用空指针或哨兵节点表示)是黑色
  • 红色节点的两个子节点必须都是黑色(即不能有两个连续的红节点)
  • 从任一节点到其每个叶子的所有路径上,包含相同数目的黑色节点(黑高一致)

关键操作:插入与修复

插入新节点默认为红色(避免直接破坏性质5),但可能违反性质4(出现红-红父子)。此时需根据父节点、叔节点颜色及位置关系分情况处理:

  • 情况1:叔节点为红 → 父、叔染黑,祖父染红,再以祖父为当前节点向上修复
  • 情况2:叔节点为黑,且当前节点是内侧插入(如父为左,当前为右) → 先对父节点左旋,转为外侧情形
  • 情况3:叔节点为黑,且为外侧插入(如父为左,当前为左) → 父染黑,祖父染红,再对祖父右旋

三种情况均能在至多两次旋转 + 若干变色后恢复红黑性质。实际编码中常将“内侧→外侧”合并为统一的旋转预处理步骤。

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

节点设计与内存管理

典型节点结构包含:keyvalue(对 map)、color(bool 或 enum)、leftrightparent 指针。STL 实现中常用 带父指针的三叉链表,便于回溯修复;部分精简实现会省略 parent,改用记录路径(但增加常数开销)。

Faceswap
Faceswap

免费开源的AI换脸工具

下载

为避免频繁 new/delete,现代实现(如 libstdc++)使用 std::allocator + 内存池管理节点,构造/析构由容器控制,不暴露裸指针给用户。

与标准库容器的对应关系

std::map 是红黑树的键值对映射,按 K 排序;std::set 是仅存 key 的集合;std::multimapstd::multiset 允许重复 key,插入时不检查等价性,仅依赖 lower_bound 定位插入位置。

所有操作时间复杂度均为 O(log n),迭代器稳定(插入删除不使原有迭代器失效,除非指向被删节点),遍历结果严格升序(基于 operator 或自定义比较器)。

不复杂但容易忽略:红黑树不是唯一选择——C++20 后部分场景可用 std::unordered_map(哈希表,平均 O(1))替代,但牺牲有序性和最坏 O(n) 查找;而红黑树提供确定性对数级性能和天然有序遍历能力,这是关联容器设计的根本权衡。

相关专题

更多
treenode的用法
treenode的用法

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

529

2023.12.01

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

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

4

2025.12.22

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

363

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

558

2023.08.10

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

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

19

2025.11.16

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

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

73

2025.09.05

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

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

24

2025.11.16

golang map原理
golang map原理

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

36

2025.11.17

ip地址修改教程大全
ip地址修改教程大全

本专题整合了ip地址修改教程大全,阅读下面的文章自行寻找合适的解决教程。

27

2025.12.26

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 0.6万人学习

Rust 教程
Rust 教程

共28课时 | 3.9万人学习

Git 教程
Git 教程

共21课时 | 2.2万人学习

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

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