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

RBTree模拟实现

一、概念

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

性质

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

问题:

如何做到最长路径<=2*最短路径?

不能连续红色+root为黑+每条路径黑结点数相同。

AVL和RBT性能对比:搜索->io

搜索/查找时:同一量级

插入/删除:

AVL树,插入删除时,因为要控制严格平衡,会进行大量旋转操作。        

二、结点的定义

三、Insert

寻找插入位置

先查找要插入的位置,_root根节点颜色默认为BLACK。

插入新结点的颜色为RED。

这是为了满足性质4,如果新结点为BLACK,会影响所有路径,相当于其它路径的黑结点数都距离目标个数缺少1个。

新结点为RED,只用满足性质3不是连续红结点即可。

则只需调整其祖先结点,并关注uncle结点颜色即可。

1、uncle存在且为红

2、uncle不存在

3、uncle存在且为黑

4、代码实现

	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 (kv.first > cur->_kv.first){parent = cur;cur = cur->_right;}else if (kv.first < cur->_kv.first){parent = cur;cur = cur->_left;}else return false;}cur = new Node(kv);cur->_col = RED; if (kv.first > parent->_kv.first){parent->_right = cur;}else{parent->_left = cur;}//每次新增newnode,要初始化它的_parent指针  三叉链cur->_parent = parent;//parent为红才需要调整while (parent && parent->_col == RED){Node* ppnode = parent->_parent;//1、uncle存在且为红//2、uncle不存在//3、uncle存在且为黑if (parent == ppnode->_left){Node* uncle = ppnode->_right;if (uncle && uncle->_col == RED){parent->_col = uncle->_col = BLACK;ppnode->_col = RED;//继续向上调整cur = ppnode;parent = cur->_parent;//没有父亲则cur为根,直接变黑即可}else if (uncle == nullptr || (uncle && uncle->_col == BLACK)){//uncle不变色,2种情况可以合成一种if (cur == parent->_left){//     pp//   p 	  //cRotateR(ppnode);parent->_col = BLACK;ppnode->_col = RED;}else{//     pp//   p 	  //		cRotateL(parent);RotateR(ppnode);cur->_col = BLACK;ppnode->_col = RED;}break;//只要旋转完就break}	}else{Node* uncle = ppnode->_left;if (uncle && uncle->_col == RED){parent->_col = uncle->_col = BLACK;ppnode->_col = RED;//继续向上调整cur = ppnode;parent = cur->_parent;//没有父亲则cur为根,直接变黑即可}else if (uncle == nullptr || uncle && uncle->_col == BLACK){//uncle不变色,2种情况可以合成一种if (cur == parent->_right){//     pp//   u    p 	  //			cRotateL(ppnode);ppnode->_col = RED;parent->_col = BLACK;}else{//     pp//   u	  p 	  //		cRotateR(parent);RotateL(ppnode);cur->_col = BLACK;ppnode->_col = RED;}break;//只要旋转完就break}}}_root->_col = BLACK;return true;}

四、IsBalance检验是否平衡

必须在满足是红黑树的条件下,检验其所有性质。

1、若简单的计算最长路径和最短路径,可能会出现连续RED的情况,不满足。

2、遍历所有路径,统计每条路径黑结点的个数,看是否都相同,遍历过程可以检查是否存在连续RED结点。

http://www.lryc.cn/news/163031.html

相关文章:

  • AUTOSAR规范与ECU软件开发(实践篇)10.4、AP和CP
  • css 命名规则
  • 正中优配:旅游餐饮板块走高,曲江文旅涨停,西安旅游等拉升
  • 世界青岛中国海洋大学金秋悦读《乡村振兴战略下传统村落文化旅游设计》2023新学年许少辉八一新书
  • 15 | Spark SQL 的 SQL API 操作
  • 为什么工作流中围绕XML做EDI报文数据解析/生成?
  • C++的运算符重载介绍
  • C++vector的使用
  • angular测试API
  • mfc 浮动窗口
  • 【C++漂流记】函数的高级应用——函数默认参数、占位参数、重载
  • Java——》synchronized的原理
  • CPU主频
  • PHP8中查询数组中指定元素-PHP8知识详解
  • 在Git中将本地分支推送到远程仓库
  • 【数据仓库基础(四)】数据仓库需求:基本需求和数据需求
  • C++类模板是一种通用的编程工具,可以创建可以适用于多种数据类型的类
  • Vite和Webpack如何使用CDN包
  • TOWE雷达光敏感应开关,让生活更智能、更安全
  • git:亲测体验rebase与merge
  • 深度神经网络之BiseNet
  • Ubantu终端常用命令、快捷键和基本操作
  • 9.5 校招 内推 面经
  • 计算机网络中的应用层和传输层(http/tcp)
  • 基于antd+vue2来实现一个简单的绘画流程图功能
  • 【小吉送书—第二期】阿里后端开发:抽象建模经典案例
  • 深度学习常用的Python库(核心库、可视化、NLP、计算机视觉、深度学习等)
  • Android菜单(上下文菜单)(选项菜单)
  • l8-d11 TCP连接管理与UDP协议
  • Python+Requests+Pytest+Excel+Allure 接口自动化测试项目实战【框架之间的对比】