数据结构--红黑树(RBTree)

一、红黑树概念

1.1 什么是红黑树

红黑树,是一种二叉搜索树,但在每个结点上增加一个存储位表示结点的颜色,可以是Red或 Black。 通过对任何一条从根到叶子的路径上各个结点着色方式的限制,红黑树确保没有一条路径会比其他路径长出俩倍,也就是最长路径不超过最短路径的2倍,因而是接近平衡的。

2.1 红黑树的性质
  1. 每个结点不是红色就是黑色
  2. 根节点是黑色的
  3. 如果一个结点是红色的,则他的两个孩子是黑色的,及不存在两个连续的红色结点
  4. 对于每个叶子结点,从该结点到其所有后代叶结点的简单路径,均包含相同数目的黑色节点
  5. 每个叶子结点都是黑色的(此处的叶子节点指的是空节点)

思考:为什么满足上面的性质,红黑树就能保证:其最长路径中节点个数不会超过最短路径节点 个数的两倍?

【解释】:由于红黑树要保证每条路径的黑色结点个数相同,且不能存在连续的两个红色结点,所以最短路径就是全黑,最长路径是一黑一红交替,这样红黑树的最长路径的极限也只能为最短路径的两倍

二、红黑树的实现

2.1 红黑树结点的定义
enum Color
{RED,BLACK
};
template<class K,class V>
struct RBTreeNode
{RBTreeNode(const pair<K,V>& kv):_left(nullptr),_right(nullptr),_parent(nullptr),_kv(kv),_col(RED){}struct RBTreeNode* _left;struct RBTreeNode* _right;struct RBTreeNode* _parent;pair<K, V> _kv;Color _col;
};

思考:在结点的定义中,为什么要将结点的默认颜色给成红色的?

【解释】:若将结点的颜色默认为黑色,由于要求每条路径的黑色结点个数相同,如果新插入一个结点,就会破坏这个性质,影响比较大,如果将结点的颜色默认为红色,那么最多也就会出现两个连续的红色结点,只会影响一条路径,只需要进行调整即可,影响比较小

2.2 红黑树的插入

1. 按照二叉搜索树的规则将结点插入到红黑树中

2. 检测插入结点后,红黑树的性质是否遭到破坏

因为新节点的默认颜色是红色,因此:如果其双亲节点的颜色是黑色,没有违反红黑树任何 性质,则不需要调整;但当新插入节点的双亲节点颜色为红色时,就违反了性质三不能有连在一起的红色节点,此时需要对红黑树分情况来讨论:

约定:cur为当前节点,p为父节点,g为祖父节点,u为叔叔节点(因为cur与p一定为红色,g一定为黑色,我们只需要对u进行讨论)

  • 情况一:cur为红,p为红,g为黑,u存在且为红(假设uncle在g的右边)

注意:此时看到的树可能是一颗完整的树,也可能是一颗子树

解决方法:

        由于出现了连续的两个红色结点,我们只能将p变为黑色,因为如果cur为新增结点,将其变为黑色会破坏性质4,而将p变为黑色,会导致g的左右子树黑色节点不同,所以需要将u也变为黑色,而g有可能只是一颗子树,为了保证与其他路径的黑色节点个数相同,还需要将g变为红色如果g为根节点的话,再将其变为黑色即可。

  • 情况二:cur为红色,p为红色,g为黑色,u不存在或者u存在且为黑色(假设u在g的右边)

说明:u的情况存在两种

1.uncle不存在,那么cur一定是新增的那个结点,因为如果不是新增结点,那么cur和p至少有一个应该为黑色,不然就违反性质3了

2.uncle存在且为黑色,那么cur原来的颜色一定为黑色,现在看到cur的颜色为红色是因为cur子树在调整过程中将cur变为了红色

【解决办法】:

  • 如果cur为p的左孩子

将p变黑,g变红,以g作为父亲结点右单旋

  • 如果cur为p的右孩子

将cur变黑,g变红,先以p为父亲节点左单旋,在以g为父亲节右单旋

ps:uncle为g左边原理同上,对于旋转不清楚的可以参考数据结构--AVL树

2.3 代码实现
bool Insert(const pair<K,V>& kv)
{if (_root == nullptr){_root = new Node(kv);_root->_col = BLACK;return true;}Node* parent = nullptr;Node* cur = _root;while (cur){if (cur->_kv.first < kv.first){parent = cur;cur = cur->_right;}else if(cur->_kv.first > kv.first){parent = cur;cur = cur->_left;}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* grandfather = parent->_parent;if (parent == grandfather->_left){Node* uncle = grandfather->_right;if (uncle && uncle->_col == RED){parent->_col = uncle->_col = BLACK;grandfather->_col = RED;//继续向上调整cur = grandfather;parent = cur->_parent;}else //uncle不存在或者uncle存在且为黑{if (cur == parent->_left){//      g//    p   u//  cRotateR(grandfather);parent->_col = BLACK;grandfather->_col = RED;}else{//      g//    p   u//      cRotateL(parent);RotateR(grandfather);cur->_col = BLACK;grandfather->_col = RED;}break;}}else{Node* uncle = grandfather->_left;if (uncle && uncle->_col == RED){parent->_col = uncle->_col = BLACK;grandfather->_col = RED;//继续向上调整cur = grandfather;parent = cur->_parent;}else //uncle不存在或者uncle存在且为黑{if (cur == parent->_right){//      g//    u   p//          cRotateL(grandfather);parent->_col = BLACK;grandfather->_col = RED;}else{//      g//    u   p//      cRotateR(parent);RotateL(grandfather);cur->_col = BLACK;grandfather->_col = RED;}break;}}}_root->_col = BLACK;return true;
}

三、红黑树的验证

红黑树的检测分为两步:

1. 检测其是否满足二叉搜索树(中序遍历是否为有序序列)

2. 检测其是否满足红黑树的性质

bool IsBalance()
{if (_root->_col == RED)return false;Node* cur = _root;int RefNum = 0; //作为路径黑色结点个数的参考while (cur){if (cur->_col == BLACK)RefNum++;cur = cur->_left;}return _IsBalance(_root,RefNum,0);
}bool _IsBalance(Node* root,int RefNum,int num)
{if (root == nullptr){//验证每条路径的黑色节点个数是否相同if (num != RefNum){cout << "黑色节点数不同" << endl;return false;}return true;}//验证是否存在两个连续的红色节点if (root->_col == RED && root->_parent->_col == RED){cout << "连续两个红色" << root->_kv.first << endl;return false;}if (root->_col == BLACK)num++;return _IsBalance(root->_left, RefNum, num) && _IsBalance(root->_right, RefNum, num);
}

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

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

相关文章

openEuler-22.03-LTS安装opengauss5.0.1(包含cm集群管理)主备

环境说明 openEuler-22.0.3-LTS opengauss5.0.1 安装数据库 安装系统依赖包 yum -y install lksctp* yum -y install psmisc yum -y install bzip2 yum -y install unzip yum -y install gcc yum -y install gcc-c yum -y install perl yum -y install libxml2-devel yum …

前端(包含cocosCreator)开发环节调取后端接口时跨域,解决办法之反向代理

/** eslint-disable */ var http require(http),httpProxy require(http-proxy),HttpProxyRules require(http-proxy-rules);// Set up proxy rules instance var port 9090 var proxyRules new HttpProxyRules({rules: {/api/(.*): https://baidu.com/$1, // 测试环境游戏…

自学VBA 设置单元格文字格式 笔记

一.设定对应单元格对应需要显示的格式 Cells(1, 1).Font.Size 18 字体大小 Cells(1, 2).Font.Color RGB(255, 0, 0) 字体颜色 Cells(1, 3).Font.Name "黑体" 字体类型 Cells(1, 4).Font.Italic True 字体斜体 Cells(1, 5).Font.FontStyle "BOLD"…

ubuntu下gcc编译器的安装

.gcc编译器的安装 一般linux下是覆盖含有的&#xff0c;如果没有执行更新命令 sudo apt update gcc安装成功&#xff0c;可以检查一下版本 可以看出我的gcc是9.4.0版本的

验证torch.nn.Conv2d

import os import cv2 import torch import numpy as np import random import cv2 as cv from matplotlib import pyplot as pltdef f_VerifyConv2D():"""验证torch.nn.Conv2d&#xff0c; 并将输入数据及权重保存到txt文件中"""x torch.randn…

SpringBoot环境隔离Profiles

前言 通常我们开发不可能只有一个生产环境&#xff0c;还会有其它的开发&#xff0c;测试&#xff0c;预发布环境等等。为了更好的管理每个环境的配置项&#xff0c;springboot也提供了对应的环境隔离的方法。 直接上干货 知识点 激活环境方法 1&#xff0c;在application…

专用设备制造业供应商收发文件,有什么专业而轻便的方式吗?

专用设备制造业的特点包括&#xff1a;门类广、跨度大、科技含量高。它主要生产的是国民经济各部门&#xff08;包括采掘、化工、冶炼、能源、医疗卫生、环保等&#xff09;所急需的重大成套设备&#xff0c;例如矿产资源井采及露天开采设备、大型火电、水电、核电成套设备、石…

教育行业文本短信VS视频短信VS语音短信哪个好?

在教育行业中&#xff0c;文本短信、视频短信和语音短信各有其优势&#xff0c;选择哪种方式更好取决于具体的应用场景和目标。 文本短信的优势在于&#xff1a; 1.简洁明了&#xff1a;能够快速、直接地传递信息&#xff0c;对于需要快速通知或提醒的场景非常适用。 …

通过内网穿透免费部署我们的springboot+vue项目 实现跟服务器一样的效果

前文讲到通过内网穿透能够实现远程访问个人电脑的静态资源。本文将讲解通过内网穿透实现远程访问本地的项目&#xff0c;实现跟部署到服务器一样的效果&#xff1a;前文链接&#xff1a;通过内网穿透实现远程访问个人电脑资源详细过程&#xff08;免费&#xff09;&#xff08;…

深度学习之卷积神经网络理论基础

深度学习之卷积神经网络理论基础 卷积层的操作&#xff08;Convolutional layer&#xff09; 在提出卷积层的概念之前首先引入图像识别的特点 图像识别的特点 特征具有局部性&#xff1a;老虎重要特征“王字”仅出现在头部区域特征可能出现在任何位置下采样图像&#xff0c…

Python 小抄

Python 备忘单 目录 1.语法和空格 2.注释 3.数字和运算 4.字符串处理 5.列表、元组和字典 6.JSON 7.循环 8.文件处理 9.函数 10.处理日期时间 11.NumPy 12.Pandas 要运行单元格&#xff0c;请按 ShiftEnter 或单击页面顶部的 Run&#xff08;运行&#xff09;。 1.语法和空格…

三种方法进行跨服务器文件传输

今天需要在一个centOS服务器上编译一个工具, 我的本地主机是ubuntu, 但是由于服务器是合规环境, 没有文件传输的接口, 也不能访问github等外网, 所以很多依赖只能下载到ubuntu然后在想办法搞到服务器上. 这种场景有三种简单有用的办法, 整理一下. 方法一: 把主机配置成http ser…

6---Linux下版本控制器Git的知识点

一、Linux之父与Git的故事&#xff1a; Linux之父叫做“Linus Torvalds”&#xff0c;我们简称为雷纳斯。Linux是开源项目&#xff0c;所以在Linux的早期开发中&#xff0c;许多世界各地的能力各异的程序员都参与到Linux的项目开发中。那时&#xff0c;雷纳斯每天都会收到许许…

用ntpdate同步时间出现问题

1. 使用ntpdate同步 [rootnode ~]# ntpdate ntp.aliyun.com4 Aug 00:07:17 ntpdate[20924]: adjust time server 203.107.6.88 offset -0.001543 sec2. 查看时间 [rootnode ~]# date Thu Aug 4 00:07:46 CST 20223. 如果时间对不上 报错信息 cna02:~ # ntpdate ntp1.aliyu…

mysql社区版最多支持多个连接并发

MySQL社区版对于并发连接数的支持并没有一个固定的上限&#xff0c;它实际上取决于多个因素&#xff0c;包括服务器的硬件配置、MySQL的配置参数以及应用程序的设计等。 硬件配置&#xff1a;服务器的CPU、内存和磁盘I/O等硬件资源会直接影响MySQL可以处理的并发连接数。例如&a…

VMware Fusion 13.5.2 for Mac 发布,产品订阅模式首个重大变更

VMware Fusion 13.5.2 for Mac 发布&#xff0c;产品订阅模式首个重大变更 适用于基于 Intel 处理器和搭载 Apple 芯片的 Mac 的桌面虚拟化软件 请访问原文链接&#xff1a;https://sysin.org/blog/vmware-fusion-13/&#xff0c;查看最新版。原创作品&#xff0c;转载请保留…

vue props接收组件数据(类型配置)

"props"接收的常见传参类型有以下几种&#xff1a;String&#xff1a;字符串类型&#xff0c;Number&#xff1a;数字类型&#xff0c;Boolean&#xff1a;布尔类型&#xff0c;Array&#xff1a;数组类型&#xff0c;Object&#xff1a;对象类型&#xff0c;Date&am…

文章解读与仿真程序复现思路——中国电机工程学报EI\CSCD\北大核心《集装箱海港级联物流-能源耦合系统协同优化方法 》

本专栏栏目提供文章与程序复现思路&#xff0c;具体已有的论文与论文源程序可翻阅本博主免费的专栏栏目《论文与完整程序》 论文与完整源程序_电网论文源程序的博客-CSDN博客https://blog.csdn.net/liang674027206/category_12531414.html 电网论文源程序-CSDN博客电网论文源…

FPGA - GTX收发器-K码 以及 IBERT IP核使用

一&#xff0c;前言 在FPGA - Xilinx系列高速收发器---GTX中详细介绍了GTX的基础知识&#xff0c;以及IP核的调用&#xff0c;下面将补充一下GTX在使用中的高速串行数据流在接收和发送时的控制与对齐&#xff08;K码&#xff09;&#xff0c;以及高速接口GTX&#xff0c;如果G…

Springboot开发 -- Postman 调试 session 验证 接口

当我们在开发Spring Boot应用时&#xff0c;经常会遇到带有Session验证的接口&#xff0c;这些接口需要用户先登录并获取到Session ID&#xff08;或称为cookie中的JSESSIONID&#xff09;&#xff0c;然后在后续的请求中携带这个Session ID来保持会话状态。下面我将以一个实际…