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

C++红黑树详解

1 红黑树的概念

红黑树,是一种二叉搜索树,但在每个结点上增加一个存储位表示结点的颜色,可以是Red
Black。 通过对任何一条从根到叶子的路径上各个结点着色方式的限制,红黑树确保没有一条路
径会比其他路径长出俩倍,因而是接近平衡的。(红黑树是接近平衡的,AVL树是绝对平衡,但两者的性能相近,AVL树的旋转比较多且复杂 ,红黑树相对简单)

红黑树是如何做到近似平衡的呢?这就与他的性质相关,具体如下:

1. 每个结点不是红色就是黑色。
2. 根节点是黑色的 。
3. 如果一个节点是红色的,则它的两个孩子结点是黑色的 。
4. 对于每个结点,从该结点到其所有后代叶结点的简单路径上,均 包含相同数目的黑色结点 。
5. 每个叶子结点都是黑色的(此处的叶子结点指的是空结点)也就是图中的NIL。

通过上述性质我们可以得出:

一条最短的路径就是只有黑节点的路径。而一条最长路径就是黑红相间的路径,这样就可以保证没有一条路径会比其他路径长出俩倍。

2.红黑树的插入

红黑树是在二叉搜索树的基础上加上其平衡限制条件,因此红黑树的插入可分为两步:
1. 按照二叉搜索的树规则插入新节点。
2.检测新节点插入后,红黑树的性质是否造到破坏。
这里重要的就是红黑树遭到破坏后如何进行调整。
1.首先我们要确保插入的节点的颜色是红色的。为什么要这样做呢? 这是因为如果插入的节点是黑色的会影响所有的路径,因为要保证每条路径上黑色节点的数目是一样的,插入的是红节点则只用看该路径是否满足红黑树的要求。
2.如果要插入的父亲节点存在且颜色是黑色的,则无需进行调整,因为满足红黑树的性质。
3.如果要插入的父亲节点存在且颜色是红色的,则需要进项调整,具体分为一下情况:

2.1.叔叔节点存在且为红

解决方式:将p,u改为黑,g改为红,然后把g当成cur,继续向上调整。

这里的a,b,c,d,e是可以变化的对应着各种复杂的情况。

等等还有很多种可能,但是他们对应的处理方式都是一样的。

2.2叔叔节点不存在或者叔叔节点存在且为黑

无论是叔叔节点不存在还是存在且为黑,他们的处理方法都是一样的,都是旋转加变色,旋转变色后就是一颗符合规则的红黑子树。这里的旋转是根据g,p,cur节点的位置进行旋转的。

3.代码实现

#pragma once
#include<iostream>
using namespace std;enum Col
{BLACK,RED
};
template<class K,class V>
struct RBTreeNode
{RBTreeNode(const pair<K,V>& kv):_parent(nullptr),_left(nullptr),_right(nullptr),_kv(kv),_col(RED){}RBTreeNode<K, V>* _parent;RBTreeNode<K, V>* _left;RBTreeNode<K, V>* _right;pair<K, V>_kv;Col _col;
};template<class K,class V>
class RBTree
{typedef RBTreeNode<K, V> Node;
public:bool insert(const pair<K, V>& kv){if (_root == nullptr){_root = new Node(kv);_root->_col = BLACK;return true;}Node* cur = _root;Node* parent = nullptr;while (cur){if (cur->_kv.first > kv.first){parent = cur;cur = cur->_left;}else if(cur->_kv.first < kv.first){parent = cur;cur = cur->_right;}else{return false;}}//找到要插入的位置 进行链接cur = new Node(kv);if (parent->_kv.first > kv.first){parent->_left = cur;}else{parent->_right = cur;}cur->_parent = parent;//进行调整 父亲节点存在且为红 才需要调整 默认插入的节点是红色的while (parent && parent->_col==RED){Node* gradeparent = parent->_parent; //gradeparent节点一定存在 否则不会进入这里 直接就插入成功了if (parent == gradeparent->_left){Node* uncle = gradeparent->_right;//叔叔节点存在且为红if (uncle && uncle->_col == RED){//变色+继续向上调整uncle->_col = parent->_col = BLACK;gradeparent->_col = RED;//向上调整 此时parent相当于curcur = gradeparent;parent = cur->_parent;}//叔叔节点不存在 or 叔叔节点存在为黑else {//旋转加变色if (cur == parent->_left){//    g//  p//cRotateR(gradeparent);parent->_col = BLACK;gradeparent->_col = RED;}else{//    g//  p//    cRotateL(parent);RotateR(gradeparent);cur->_col = BLACK;gradeparent->_col = RED;}break;}}else{Node* uncle = gradeparent->_left;//叔叔存在且为红if (uncle && uncle->_col == RED){//变色+向上调整uncle->_col = parent->_col = BLACK;gradeparent->_col = RED;cur = gradeparent;parent = cur->_parent;}//叔叔不存在 or 叔叔存在且为黑else{//旋转+变色if (cur == parent->_right){//g//  p//    cRotateL(gradeparent);parent->_col = BLACK;gradeparent->_col = RED;}else{//g//  p//cRotateR(parent);RotateL(gradeparent);cur->_col = BLACK;gradeparent->_col = RED;}break;}}}_root->_col = BLACK;return true;}void RotateR(Node* parent){Node* cur = parent->_left;Node* curRight = cur->_right;Node* ppNode = parent->_parent;//进行链接parent->_left = curRight;if (curRight)curRight->_parent = parent;cur->_right = parent;parent->_parent = cur;if (ppNode == NULL){_root = cur;cur->_parent = nullptr;}else{if (ppNode->_left == parent){ppNode->_left = cur;}else{ppNode->_right = cur;}cur->_parent = ppNode;}}void RotateL(Node* parent){Node* cur = parent->_right;Node* curLeft = cur->_left;Node* ppNode = parent->_parent;parent->_right = curLeft;if (curLeft){curLeft->_parent = parent;}cur->_left = parent;parent->_parent = cur;if (ppNode == nullptr){_root = cur;cur->_parent = nullptr;}else{if (ppNode->_left == parent){ppNode->_left = cur;}else{ppNode->_right = cur;}cur->_parent = ppNode;}}bool CheckColour(Node* root, int blacknum, int benchmark){if (root == nullptr){if (blacknum != benchmark)return false;return true;}if (root->_col == BLACK){++blacknum;}if (root->_col == RED && root->_parent && root->_parent->_col == RED){cout << root->_kv.first << "出现连续红色节点" << endl;return false;}return CheckColour(root->_left, blacknum, benchmark)&& CheckColour(root->_right, blacknum, benchmark);}bool IsBalance(){return IsBalance(_root);}bool IsBalance(Node* root){if (root == nullptr)return true;if (root->_col != BLACK){return false;}// 基准值int benchmark = 0;Node* cur = _root;while (cur){if (cur->_col == BLACK)++benchmark;cur = cur->_left;}return CheckColour(root, 0, benchmark);}private:Node* _root = nullptr;
};
#include"RBTree.h"int main()
{int a[] = { 4, 2, 6, 1, 3, 5, 15, 7, 16, 14 };RBTree<int, int> t;for (auto e : a){t.insert(make_pair(e, e));cout << "Insert:" << e << "->" << t.IsBalance() << endl;}return 0;
}

http://www.dtcms.com/a/394386.html

相关文章:

  • 【不背八股】15.kmp算法/ Dijkstra算法/二叉树遍历
  • 【ES】ElasticSearch 数据库之查询操作 从入门>实践>精通 一篇文章包含ES的所有查询语法
  • huggingface-cli修改模型下载路径
  • 计算机视觉——灰度分布
  • OpenFeature 标准在 ABP vNext 的落地
  • Agentic AI 多智能体协作:开发实战、框架选型与踩坑指南
  • [优选算法专题三二分查找——NO.17二分查找]
  • 一文学会c++哈希
  • 【06】EPGF 架构搭建教程之 本地环境管理工具的本地化
  • 【开发实践】DNS 报文分析与 CDN 架构可视化方案
  • Ubuntu 系统下 Nginx + PHP 环境搭建教程
  • AI 如何改变日常生活
  • 字典树 Trie 介绍、实现、封装与模拟 C++STL 设计
  • 第一性原理(First Principles Thinking)
  • 1.UE-准备环境(一)-账号注册和打开虚幻引擎源码页面
  • javascript `AbortController`
  • 时间复杂度与空间复杂度
  • rocketmq队列和消费者关系
  • RAG评估指南:从核心指标到开源框架,打造高效检索生成系统
  • xtuoj 0x05-A 前缀和
  • 防误删 (实时) 文件备份系统 (btrfs 快照 + rsync)
  • 【FreeRTOS】第七课(1):任务间通信(使用队列)
  • OD C卷 - 二叉树计算
  • DiffDock 环境安装和使用教程
  • NVIC中的不可屏蔽中断NMI(Non-Maskable Interrupt)是什么?
  • TypeORM 浅析
  • 2.4 死锁 (答案见原书 P165)
  • 算法与数据结构:常见笔试题总结
  • trae使用playwright MCP方法流程
  • anaconda安装tensorflow遇到的一个错误