红黑树(RBTree)认识总结

news2024/5/19 19:03:37

一、认识红黑树

1.1 什么是红黑树?

红黑树是一种二叉搜索树,与普通搜索树不同的是,在每个节点上增加一个“颜色”变量 —— RED / BLACK 。

通过对各个节点颜色的限制,确保从 根 到 NIL ,没有一条路径会比其他路径长出两倍。

NIL :表示叶子节点的空指针,统一设置为 BLACK

1.2 红黑树的性质
  • 根节点一定是黑色
  • 不能出现两个连续的红色节点
  • 对于同一高度而言,从根到该高度任一节点的简单路径上的黑色节点的数量相同
1.3 红黑树节点定义

二、红黑树

2.1 红黑树定义
template<class K, class V>
class RBTree
{
    typedef RBTreeNode<K, V> Node;
    
private:
    Node* _root;
};
2.2 插入

红黑树的插入是我们学习红黑树过程最重要的知识之一,它主要分为两部分:平衡二叉树的插入 和 旋转 —— 调整树形结构。

插入部分与普通搜索树没有本质区别,这里不做过多介绍。

声明一下:代码中的 grandfather 和 图中的 grandparent 为同一东西,笔者在基本结束本篇时发现这里差异。

  • 插入部分
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 (cur == parent->_left)
    {
        parent->_left = cur;
    }
    else 
    {
        parent->_right = cur;
    }
    cur->_parent = parent;
    
    _root->_col = BLACK; // 强制设定根一定为黑色!
    return true;
}
  • 旋转
2.2.1 什么时候要旋转?

我们新插入的节点默认是红色,当它的 parent 存在且为红色时,就出现了这种情况 —— 树存在两个连续的红色节点,此时我们需要对该部分子树进行旋转 —— 调整树的结构。(下图只展示了部分的子树)

判断条件:parent 存在且为红色

	while (parent && parent->_col == RED)
    { }
2.2.2 几种旋转情况
情形一:uncle 存在,且为红色节点

在这里插入图片描述

	Node* grandfather = parent->_parent;
	if (parent == grandfather->_left)
    {
        Node* uncle = grandfather->_right;
        if (uncle && uncle->_col == RED) // 叔叔存在且为红色
        {
            grandfather->_col = RED;
            parent->_col = BLACK;
            uncle->_col = BLACK;

            cur = grandfather; //  向上调整
            parent = cur->_parent;
        }
    }
	
	if (parent == grandfather->_right)
    {
        Node* uncle = grandfather->_left;
        // ... // 与上面代码一致
    }
情形二:uncle 不存在 或 存在且为黑色
  • parent 在 grandfather 左侧的两种情况
	if (parent == grandfather->_left) // parent 在 grandfather 左侧的两种情况
    {
        if (!uncle || uncle->_col == BLACK)
        {
            if (cur == parent->_left) // cur 在 parent 左侧
            {
                RotateR(grandfather);
                
                parent->_col = BLACK;
                grandfather->_col = RED;
            }
            else // cur 在 parent 右侧
            {
                RotateL(parent);
                RotateR(grandfather);
                
                parent->_col = BLACK;
                grandfather->_col = RED;
            }
            break; // 旋转结束后,一定要 break 
            
        }
    }

旋转结束后,树的结构已经满足了红黑树的标准,如果不跳出循环、继续调整,会出现各种奇怪的问题。

  • parent 在 grandfather 右侧
	if (parent == grandfather->_right) // parent 在 grandfather 右侧
    {
        if (!uncle || uncle->_col == BLACK) // uncle 不存在 或 uncle存在且为黑色节点
        {
            if (cur == parent->_right) // cur 在 parent 右侧
            {
                RotateL(grandfather);
                
                parent->_col = BLACK;
                grandfather->_col = RED;
            }
            else // cur 在 parent 左侧
            {
                RotateR(parent);
                RotateL(grandfather);
                
                parent->_col = BLACK;
                grandfather->_col = RED;
            }
            break;
        }
    }

在这里插入图片描述

2.3 红黑树的验证

红黑树的验证,顾名思义,就是验证 你的“红黑树” 是否能满足红黑树的三条性质。

  • 根节点一定是黑色
  • 不能出现两个连续的红色节点
  • 对于同一高度而言,从根到该高度任一节点的简单路径上的黑色节点的数量相同
	bool IsBalance()
    {
        if (_root && _root->_col == RED) // 验证第一条性质
        {
            cout << "根节点为红色" << endl;
            return false;
        }
        
        // 要判断是否每一条路径上的黑色节点数相同,首先要找一个标杆 —— 这里旋转树最左路径的黑色节点个数
        Node* cur = _root;
        int RefBlackNum = 0;
        while (cur)
        {
            if (cur->_col == BLACK)
                ++RefBlackNum;
            
            cur = cur->_left;
        }
        
        return Check(_root, 0, RefBlackNum);
    }
	bool Check(Node* cur, int BlackNum, int RefBlackNum)
    {
        if (cur == nullptr) //  走到 NIL 时,判断该路径黑色节点个数是否与标杆相同
        {
            if (BlackNum != RefBlackNum)
            {
                cout << "路径黑色节点的个数不相同" << endl; // 验证第三条性质
                return false;
            }
            return true;
        }
        
        if (cur->_col == RED && cur->_parent && cur->_parent->_col == RED)
        {
            cout << "存在两个连续的红色节点" << endl; // 验证第二条性质
            return false;
        }
        
        if (cur->_col == BLACK)
            ++BlackNum;
        
        return Check(cur->_left, BlackNum, RefBlackNum) // 递归判断当前节点的左右子树是否合法
            && Check(cur->_right, BlackNum, RefBlackNum);
    }

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1648235.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

论文| Visual place recognition: A survey from deep learning perspective

2021-Visual place recognition: A survey from deep learning perspective

anconda创建虚拟环境,使用虚拟环境(基于win平台)

假设已经安装了anconda&#xff0c;打开anaconda的 shell。 查看已存在的虚拟环境&#xff0c;base是默认的&#xff0c;不用理会&#xff0c;后面的yolov5就是用户创建的 #查看有那些虚拟环境 (base) PS C:\Users\x> conda info -e # conda environments: # base …

hadoop---基于Hive的数据仓库相关函数机制及其优化方案

Hive相关函数&#xff08;部分&#xff09;&#xff1a; if函数: 作用: 用于进行逻辑判断操作 语法: if(条件, true返回信息,false返回信息) 注意: if函数支持嵌套使用 select if(aa,’bbbb’,111) fromlxw_dual; bbbb select if(1<2,100,200) fromlxw_dual; 200nvl函数:…

Windows平台下Redis通过配置文件解决密码重置问题

Reids会在每次重启服务器后重置通过指令设置的密码&#xff0c;通过修改配置文件中的密码则可以避免这一问题&#xff0c;保证密码的永久有效性。&#xff08;注意点&#xff1a;博主已经配置好了Redis的系统环境变量&#xff0c;所以不用进入到Redis目录下&#xff0c;也能直接…

MYSQL RR隔离级别下无索引更新是否表锁?

最近在MYSQL菜鸟群有群友提问,说他看了某个公众号里面文章说 "MYSQL RR隔离级别下无索引更新会导致表锁! " 他表示疑惑,而且不仅是他,还有很多个她在不同的群里同样表示疑惑! 下面是群友的截图 是啊 MYSQL 以及进化到了8.3.0版本了,普遍都使用5.7和8.0版本.而且还…

运营商实名认证API接口如何对接

运营商实名认证API接口又叫手机号实名认证API接口&#xff0c;根据入参字段不同&#xff0c;分为两类。一类是运营商手机二要素核验API接口&#xff0c;一类是运营商手机三要素核验API接口。运营商手机二要素验证API接口指的是传入姓名及手机号码&#xff0c;通过运营商权威渠道…

FX95GT FX505GT windows 11 触摸板安装

FX95GT FX505GT windows 11 触摸板驱动安装 如果正常使用 exe 文件安装不上&#xff0c;请在 ‘设置’ 》 ‘系统信息 ’》 驱动下载地址 如果正常使用 exe 文件安装不上&#xff0c;请在 ‘设置’ 》 ‘系统信息 ’》 高级系统设置 设备管理 在电脑上点右键&#xff0c;选择…

合肥先进光源束测步进电机控制机箱接线方式

合肥先进光源束测步进电机控制方案介绍 对上篇文里的接线方式做个修订&#xff1a; EtherCat电机控制机箱接线规范 驱动器 控制器 接线方式 使用鸣志 STF05-ECX-H驱动器&#xff0c;每个机箱配8个驱动器使用汇川的H5U-1614MTD etherCat控制器每个驱动器的电源从2分8的分配端子…

GitHub搭建免费博客

一、GitHub仓库准备 ​ 搭建博客需要准备两个仓库。一个存放博客图床的仓库&#xff0c;另一个存放博客网站的仓库。 1.1、图床创建 新建仓库 第一步&#xff1a; ​ 第二步&#xff1a; 生成Token令牌 点击右上角头像->Settings->下拉&#xff0c;直到左侧到底&#…

Tmux工具使用案例

Tmux工具使用案例 连接linux一般使用ssh&#xff0c;当ssh会话中需要长时间执行命令时&#xff0c;为了避免命令不受ssh会话影响&#xff0c;除了可以将命令通过nohup <cmd> &等方法放到后台执行外&#xff0c;也可以利用Tmux这个工具解绑SSH会话与执行命令&#xff…

【Python文本数据系列】使用TextCNN模型进行文本情感分析(案例+源码)

这是我的第273篇原创文章。 一、引言 当使用深度学习处理文本分类任务时&#xff0c;遵循的流程如下&#xff1a; 首先&#xff0c;准备数据&#xff0c;而且数据量要大&#xff0c;才能发挥神经网络的优势&#xff1b; 再者&#xff0c;进行文本预处理&#xff0c;将文本数据…

Adobe-Premiere-CEP 扩展 入门-视频剪辑-去气口插件-Silence Remover

短视频&#xff0c;这两年比较火&#xff0c;不要再问为什么用Premiere&#xff0c;非常难用&#xff0c;为什么不用某影&#xff0c;某些国内软件非常接地气简单&#xff0c;又例如某音资深的视频短编辑就很好用了。。。 Premiere二次开发调试难&#xff0c;不如自己搞个cons…

(41)5.6-5.7数据结构(栈和队列的应用)

1.栈在括号匹配中的应用 #define _CRT_SECURE_NO_WARNINGS #define MaxSize 10 typedef struct { char data[MaxSize];//静态数组存放栈中元素 int top; //栈顶指针 }SqStack;//初始化栈 void InitStack(SqStack& S);//判断栈是否为空 bool StackEmpty(SqStack S…

python:机器学习特征优选

作者&#xff1a;CSDN _养乐多_ 在Python中进行机器学习特征选择的方法有很多种。以下是一些常用的方法&#xff1a; 过滤法&#xff08;Filter Methods&#xff09;&#xff1a;通过统计方法或者相关性分析来评估每个特征的重要性&#xff0c;然后选择最相关的特征。常用的…

Obsidian dataview 使用入门

Dataview有四种展示格式&#xff1a;list、table、task、calendar。 本文只介绍前面两种。 语法总结 通过#标签 dataview LIST FROM #标签 通过"文件夹" dataview LIST FROM "文件夹名" 通过[ [ 文件链接 ] ] 选择链接到一个文件&#xff0c;或者…

​分享1.36G全国村名点数据

数据是GIS的血液&#xff01; 我们在《2015年中国电子地图数据》一文中&#xff0c;为大家有偿分享了一份图层丰富&#xff0c;且有26.8G大小的全国电子地图。 这里再为大家分享一份有1.36G大小的全国村名数据&#xff0c;本数据来自网友分享&#xff0c;据说为2023年的村名数…

1688 API集成,智能选品铺货Fecify,多语言支持,全球畅销无忧!

跨境独立站铺货&#xff0c;一般都是无库存模式&#xff0c;大致思路&#xff1a;卖家没有进行商品的采购囤货&#xff0c;先采集商品数据&#xff0c;发布到独立站&#xff0c;推广引流出单后&#xff0c;用订单商品的图片&#xff0c;去1688用图片搜商品&#xff0c;采购订单…

新书速览|图神经网络基础、模型与应用实战

掌握PyTorch图神经网络基础与模型&#xff0c;实战自然语言处理、计算机视觉、推荐系统、社交网络应用开发 01 本书内容 图神经网络不仅能够解决传统机器学习方法无法解决的图数据问题&#xff0c;而且能够应用于许多实际场景&#xff0c;例如社交网络、药物发现、网络安全、…

Redis是什么? 日常运维 Redis 需要注意什么 ? 怎么降低Redis 内存使用 节省内存?

你的项目或许已经使用 Redis 很长时间了&#xff0c;但在使用过程中&#xff0c;你可能还会或多或少地遇到以下问题&#xff1a; 我的 Redis 内存为什么增长这么快&#xff1f;为什么我的 Redis 操作延迟变大了&#xff1f;如何降低 Redis 故障发生的频率&#xff1f;日常运维…

VMP 简单源码分析(.net)

虚拟机 获取CPU的型号 实现了一个指令集解释器&#xff0c;每个操作码对应一个特定的处理函数&#xff0c;用于执行相应的指令操作。在执行字节码时&#xff0c;解释器会根据操作码查找并调用相应的处理函数来执行指令。 截获异常 先由虚拟机处理 处理不了再抛出异常 priva…