【数据结构】红黑树详解

目录

前言:

红黑树的概念:

红黑树的性质:

红黑树节点的定义:

红黑树的插入:

情况1:cur为红,p为红,g为黑,u存在且为红 

情况2:cur为红,p为红,g为黑,u不存在或者u为黑(p和cur都在其父亲节点同一侧)

情况3:cur为红,p为红,g为黑,u不存在或者u为黑(p和cur在其父亲节点的不同侧)

红黑树验证:

AVL树和红黑树的比较:

红黑树应用:

结语:


前言:

如果有需要文章源码的友友请前往:红黑树源码

找到RBTree即红黑树源码。

红黑树的概念:

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

下图就是一颗红黑树。

红黑树的性质:

(1)最长路径最多是最短路径的2倍。

(2)每个结点不是红色就是黑色。

(3)根节点是黑色的 。

(4) 如果一个节点是红色的,则它的两个孩子结点是黑色的(没有2个连续的红色节点) 

(5)对于每个结点,从该结点到其所有后代叶结点的简单路径上,均包含相同数目的黑色结点。

(6)每个叶子结点都是黑色的(此处的叶子结点指的是空结点)。

红黑树节点的定义:

红黑树采用孩子双亲表示法,多了一个表示颜色的枚举类(只有BLACK和RED).

public class RBTreeNode {public int val;public RBTreeNode left;public RBTreeNode right;public RBTreeNode parent;public COLOR color;public RBTreeNode(int val){this.val = val;color = COLOR.RED;}
}

其中COLOR是枚举类型具体代码如下: 

public enum COLOR{RED,BLACK
}

在看到上面红黑树的构造方法后,不知道友友们有没有对把新节点设置成红色感到疑惑尼? 

提问:在节点的定义中,为什么要将节点的默认颜色给成红色的?

答:这是因为根据上面给出的性质(5),红黑树从某节点到其后代叶节点的路径上的黑色节点个数相同,那么这个时候如果直接插入一个黑色节点,那么为了维护性质(5)必须要在别的叶节点后面也插入黑色节点,但是我们本意是插入一个节点,故不能插入黑色节点。至于插入红色节点可能会违反性质(4)我们只要调节一下颜色即可。

红黑树的插入:

红黑树是在二叉搜索树的基础上,加上其平衡限制条件,因此红黑树的插入可分为两步:

(1)按照二叉搜索的树规则插入新节点。

(2)检测新节点插入后,红黑树的性质是否造到破坏。

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

在这之前我们先来个规定🙌🙌🙌:

规定:cur为当前节点,p为父节点,g为祖父节点,u为叔叔节点。

总共有三种情况具体如下:

情况1:cur为红,p为红,g为黑,u存在且为红

这里解释的cur在p的左边如果cur是在p的右边的情况可以把三种情况都看完,相信就明白啦👍👍👍

如下图:

那么我们要怎么调整才能使它符合红黑树的性质呢?

具体调整过程如下:

注意:这里看到的树,可能是一个完整的树,也可能是一颗子树。

(1)如果g是根节点,调整完成后,需要将g改成黑色。

(2)如果g是子树,g一定有双亲,且g的双亲如果是红色,需要继续向上调整。

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

情况2:cur为红,p为红,g为黑,u不存在或者u为黑(p和cur都在其父亲节点同一侧)

说明:情况2可以根据u存不存在分为两种:

(1)如果u节点不存在,则cur一定是新插入节点(第二种情况的cur原本是黑色的,因为向上调整变成红色),因为如果cur不是新插入节点,则cur和p一定有一个节点的颜色是黑色,这样就不满足性质(5):每条路径黑色节点个数相同。

(2)如果u节点存在,则其一定是黑色(情况规定),那么cur节点原来的颜色一定是黑色的,现在看到其是红色的原因是因为cur的子树在调整的过程中将cur节点的颜色由黑色变成红色。

第1种类型如下图:

如果对旋转不太清楚的话可以看看我的上一篇博客实现的很清楚了。AVL树

先进行把g右旋,在把p和g的颜色改一下就可以弄出符合6条性质的红黑树。 

第2种类型如下图:

第2种类型是在调整的过程中才会出现的如下图。

 出现第2中类型后我们先把g右旋,接着把p变黑,g变红。

情况2的总结:

p为g的左孩子,cur为p的左孩子,则进行右单旋转。

p为g的右孩子,cur为p的右孩子,则进行左单旋转。

p、g变色--》p变黑,g变红。

情况3:cur为红,p为红,g为黑,u不存在或者u为黑(p和cur在其父亲节点的不同侧)

单看名字可能觉得和情况2差不多,其实就是多了一层旋转,如果有学过AVL树的友友不难发现这其实和AVL树的双旋差不多(没学过也没关系啦👍👍👍)。

当u不存在的情况下:

我们可以看到先将p节点左旋,再把p和cur的指向互换,这样就变成情况2,接着我们按照情况2来处理即可。

当u存在的情况下:

先根据情况1调整接着把p节点左旋,并交换p和cur的指向。最后按照情况2的处理方式即可。

到这里我们插入的三种情况终于全部讲解完毕🎉🎉🎉 

注意:我们上面分析的三种情况都是parent在grandFather的左侧,由于文章篇幅有限且parent在grandFather的右侧分析和在左侧是一模一样的,故这里就不再过多赘述,画完图后我们就会发现parent在grandFather的右侧就是和在左侧相反的情况,即把代码中出现left的改为right,right改为left即可(旋转方法也是)。 

插入的具体源码如下:

旋转源码因为和上一章AVL树差不多如果有需要源码的友友在文章的开头有给出😎😎😎。

public boolean insert(int val) {//1.按照二叉搜索树的方式插入节点RBTreeNode node = new RBTreeNode(val);if (root == null) {root = node;root.color = COLOR.BLACK;return true;}RBTreeNode parent = null;RBTreeNode cur = root;//记得要及时更新parent节点while (cur != null) {if (val > cur.val) {parent = cur;cur = cur.right;} else if (val < cur.val) {parent = cur;cur = cur.left;} else {System.out.println("存在重复节点");return false;}}if (val < parent.val) {parent.left = node;} else {parent.right = node;}//千万要记得我们还要把node的parent指回去。node.parent = parent;cur = node;//进行红黑树的调整while (parent != null && parent.color == COLOR.RED) {//注意进入这个while循环这里的grandFather是必定存在的,//因为parent为红色,但是我们的根节点只能是黑色RBTreeNode grandFather = parent.parent;if (grandFather.left == parent) {RBTreeNode uncle = grandFather.right;if (uncle != null && uncle.color == COLOR.RED) {parent.color = COLOR.BLACK;uncle.color = COLOR.BLACK;grandFather.color = COLOR.RED;cur = grandFather;parent = cur.parent;} else {//uncle不存在或者uncle为黑色//经过分别画图uncle不存在和uncle为黑色的情况分析发现处理方式是一样的//只要区分cur在parent的左侧还是右侧即可//和AVL树的思路差不多先旋转预处理一下,再来一次旋转搞定。//即转化为cur和parent在同一侧的情况。if (cur == parent.right) {rotateLeft(parent);RBTreeNode tmp = parent;parent = cur;cur = tmp;}//转换成一侧的情况rotateRight(grandFather);grandFather.color = COLOR.RED;parent.color = COLOR.BLACK;}} else {//grandFather.right == parent//这里就只要把上面的代码对应反着来就行//就是把上面代码里面出现right变成left,left变成right,方法也要变RBTreeNode uncle = grandFather.left;if (uncle != null && uncle.color == COLOR.RED) {parent.color = COLOR.BLACK;uncle.color = COLOR.BLACK;grandFather.color = COLOR.RED;cur = grandFather;parent = cur.parent;} else {//uncle不存在或者uncle为黑色//经过分别画图uncle不存在和uncle为黑色的情况分析发现处理方式是一样的//只要区分cur在parent的左侧还是右侧即可//和AVL树的思路差不多先旋转预处理一下,再来一次旋转搞定。//即转化为cur和parent在同一侧的情况。if (cur == parent.left) {rotateRight(parent);RBTreeNode tmp = parent;parent = cur;cur = tmp;}//转换成一侧的情况rotateLeft(grandFather);grandFather.color = COLOR.RED;parent.color = COLOR.BLACK;}}}root.color = COLOR.BLACK;return true;}

红黑树验证:

写完代码那就肯定要验证一下下啦❤️❤️❤️

红黑树的检测分为两步:

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

(2)检测其是否满足红黑树的性质。

中序遍历代码简单我就不贴了。

空树也是红黑树。

根节点的颜色是黑色。

判断没有两个连续的红色节点(checkRedNode)和每条路径上黑色节点数相同(checkBlackNum)

public boolean isValidRBTree(){//1.红的节点后面只能是黑色节点//2.所有节点到其后代叶子节点的路径上的黑色节点个数相同//3.根节点必须是黑色节点if(root == null){return true;}if(root.color != COLOR.BLACK){System.out.println("违反了根节点必须是黑色的特性");return false;}int tempBlackNum = 0;RBTreeNode cur = root;while(cur != null){if(cur.color == COLOR.BLACK){tempBlackNum++;}cur = cur.left;}return checkRedNode(root) && checkBlackNum(root,0,tempBlackNum);}

要想确定每一条路径上黑色节点个数相同那么我们就要先找一条路径来作参照,至于正不正确我们不关心只要保证黑色节点数相同即可。

找参照黑色节点数在外面找这里看上面代码我找的是最左边的那条路。

checkBlackNum源码如下:

private boolean checkBlackNum(RBTreeNode root,int path,int tempBlackNum){if(root == null){return true;}//要先加在判断,根据实际情况来if(root.color == COLOR.BLACK){path++;}if(root.left == null && root.right == null){if(path != tempBlackNum){System.out.println("违反了每条路径上的黑色节点数是一样的");return false;}}return checkBlackNum(root.left,path,tempBlackNum) && checkBlackNum(root.right,path,tempBlackNum);}

checkRedNode源码如下:

这里root的颜色如果是红色的话那么其父亲节点必定存在,因为根节点是黑色的所以不用判断是否为空。可以看到在树的各类方法中递归是非常参见的,写递归时我们关注某一个子问题做什么就好了。

private boolean checkRedNode(RBTreeNode root){if(root == null){return true;}if(root.color == COLOR.RED){if(root.parent.color == COLOR.RED){System.out.println("违反了两个红色节点不能连续出现");return false;}}return checkRedNode(root.left) && checkRedNode(root.right);}

这里也建议大家拿这个这个测试用例去跑一跑。

不仅要返回true还要调试一遍看看和自己手动画的图是不是一样的,下面我给出测试用例的红黑树的图,也希望友友们能自己画出来。 

红黑树的删除本文章不做讲解,有兴趣的同学可参考:《算法导论》

作者准备学了(希望不要入门到入土😭😭😭),后续有需要的话可能会单独出一篇👌👌👌。

AVL树和红黑树的比较:

红黑树和AVL树都是高效的平衡二叉树,增删改查的时间复杂度都是O(logN),红黑树不追求绝对平衡,其只需保证最长路径不超过最短路径的2倍,相对而言,降低了插入和旋转的次数,所以在经常进行增删的结构中性能比AVL树更优,而且红黑树实现比较简单,所以实际运用中红黑树更多。

红黑树应用:

1. java集合框架中的:TreeMap、TreeSet底层使用的就是红黑树。

2. C++ STL库 -- map/set、mutil_map/mutil_set。

3. linux内核:进程调度中使用红黑树管理进程控制块,epoll在内核中实现时使用红黑树管理事件块。

4. 其他一些库:比如nginx中用红黑树管理timer等。

结语:

其实写博客不仅仅是为了教大家,同时这也有利于我巩固知识点,和做一个学习的总结,由于作者水平有限,对文章有任何问题还请指出,非常感谢。如果大家有所收获的话还请不要吝啬你们的点赞收藏和关注,这可以激励我写出更加优秀的文章。

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

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

相关文章

【Java EE】关于Maven

文章目录 &#x1f38d;什么是Maven&#x1f334;为什么要学Maven&#x1f332;创建⼀个Maven项目&#x1f333;Maven核心功能&#x1f338;项目构建&#x1f338;依赖管理 &#x1f340;Maven Help插件&#x1f384;Maven 仓库&#x1f338;本地仓库&#x1f338;私服 ⭕总结 …

电商平台混战之下,天猫破解品牌增长奥秘

行业共识是追上风&#xff0c;才有好生意&#xff0c;但风很多时候不会只有一个方向。 4月2日&#xff0c;上海&#xff0c;TopTalk 2024天猫超级品牌私享会举行。这个活动已举办数年&#xff0c;每一年天猫都会发布新一年度的品牌经营策略&#xff0c;只是与往年不同的是&…

从零开始学起!全方位解析App压力测试的关键要点!

简介 Monkey 是 Google 提供的一个用于稳定性与压力测试的命令行工具 可以运行在模拟器或者实际设备中 它向系统发送伪随机的用户事件对软件进行稳定性与压力测试 为什么要用 Monkey Monkey 就是像猴子一样上蹿下跳地乱点 为了测试软件的稳定性&#xff0c;健壮性 随机点…

区间概率预测python|QR-CNN-BiLSTM+KDE分位数-卷积-双向长短期记忆神经网络-时间序列区间概率预测+核密度估计

区间预测python|QR-CNN-BiLSTMKDE分位数-卷积-双向长短期记忆神经网络-核密度估计-回归时间序列区间预测 模型输出展示&#xff1a; (图中是只设置了20次迭代的预测结果&#xff0c;宽度较宽&#xff0c;可自行修改迭代参数&#xff0c;获取更窄的预测区间&#xff09; 注&am…

2012年认证杯SPSSPRO杯数学建模D题(第一阶段)人机游戏中的数学模型全过程文档及程序

2012年认证杯SPSSPRO杯数学建模 D题 人机游戏中的数学模型 原题再现&#xff1a; 计算机游戏在社会和生活中享有特殊地位。游戏设计者主要考虑易学性、趣味性和界面友好性。趣味性是本质吸引力&#xff0c;使玩游戏者百玩不厌。网络游戏一般考虑如何搭建安全可靠、丰富多彩的…

汇编语言:寻址方式在结构化数据访问中的应用——计算人均收入

有一年多没有在CSDN上发博文了。人的工作重心总是有转移的&#xff0c;庆幸一直在做着有意义的事。   今天的内容&#xff0c;是为汇编语言课程更新一个实验项目。      本方案修改自王爽编《汇编语言》第&#xff14;版P172“实验7寻址方式在结构化数据访问中的应用” …

微软推出GPT-4 Turbo优先使用权:Copilot for Microsoft 365商业用户享受无限制对话及增强图像生成能力

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

【Apache Doris】周FAQ集锦:第 1 期

【Apache Doris】周FAQ集锦&#xff1a;第 1 期 SQL问题数据操作问题运维常见问题其它问题关于社区 欢迎查阅本周的 Apache Doris 社区 FAQ 栏目&#xff01; 在这个栏目中&#xff0c;每周将筛选社区反馈的热门问题和话题&#xff0c;重点回答并进行深入探讨。旨在为广大用户和…

【单】Unity _RPG项目中的问题

&#x1f468;‍&#x1f4bb;个人主页&#xff1a;元宇宙-秩沅 &#x1f468;‍&#x1f4bb; hallo 欢迎 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! &#x1f468;‍&#x1f4bb; 本文由 秩沅 原创 &#x1f468;‍&#x1f4bb; 收录于专栏&#xff1a; ⭐…

【Error】Uncaught TypeError: Cannot read properties of undefined (reading ‘get’)

报错原因&#xff1a; 返回值为undefined 解决&#xff1a; vue3可用&#xff1f;

echarts实现炫酷科技感的流光效果

前言&#xff1a; echarts实现炫酷科技感的流光效果 效果图&#xff1a; 实现步骤&#xff1a; 1、引入echarts,直接安装或者cdn引入 npm i echarts https://cdn.jsdelivr.net/npm/echarts5.4.3/dist/echarts.min.js 2、封装 option方法&#xff0c;第一个数据是折线数据&a…

面试:HashMap

目录 1、底层数据结构&#xff0c;1.7 与1.8有何不同? 2、为何要用红黑树&#xff0c;为何一上来不树化&#xff0c;树化阈值为何是8&#xff0c;何时会树化&#xff0c;何时会退化为链表? 3、索引如何计算? hashCode都有了&#xff0c;为何还要提供hash()方法?数组容量为…

AJAX —— 学习(三)(完结)

目录 一、jQuery 中的 AJAX &#xff08;一&#xff09;get 方法 1.语法介绍 2.结果实现 &#xff08;二&#xff09;post 方法 1.语法介绍 2.结果实现 &#xff08;三&#xff09;通用型的 AJAX 方法 1.语法介绍 2.结果实现 二、AJAX 工具库 axios &#xff08…

【XCPC笔记】2023 (ICPC) Jiangxi Provincial Contest——ABCHIJKL 做题记录

赛后gym练习及补题&#xff0c;gym链接&#xff1a;2023 (ICPC) Jiangxi Provincial Contest – Official Contest 补题顺序 L [Zhang Fei Threading Needles - Thick with Fine](https://codeforces.com/gym/104385/problem/L)题面解读参考代码 A [Drill Wood to Make Fire](h…

【数据结构与算法】力扣 24. 两两交换链表中的节点

题目描述 给你一个链表&#xff0c;两两交换其中相邻的节点&#xff0c;并返回交换后链表的头节点。你必须在不修改节点内部的值的情况下完成本题&#xff08;即&#xff0c;只能进行节点交换&#xff09;。 示例 1&#xff1a; 输入&#xff1a; head [1,2,3,4] 输出&#…

迷茫下是自我提升

长夜漫漫&#xff0c;无心睡眠。心中所想&#xff0c;心中所感&#xff0c;忧愁当前&#xff0c;就执笔而下&#xff0c;写下这篇文章。 回忆过往 回想当初为啥学前端&#xff0c;走前端这条路&#xff0c;学校要求嘛&#xff0c;兴趣爱好嘛&#xff0c;还是为了钱。 时间带着…

使用GPT需要注意的事项

GPT出来之后&#xff0c;基本就告别浏览器搜索问题答案了。将问题原封不动的copy给GPT基本可以得到解答。 但是这个也有弊端&#xff0c;那就是太依赖GPT了。 1&#xff0c;使用GPT需要更强的专业知识&#xff1a;除了能问对问题&#xff0c;还要具备识别GPT&q…

WordPress 6.5 “里贾纳”已经发布

WordPress 6.5 “里贾纳”已经发布&#xff0c;其灵感来自著名爵士小提琴家Regina Carter的多才多艺。雷吉娜是一位屡获殊荣的艺术家和著名的爵士乐教育家&#xff0c;以超越流派而闻名&#xff0c;她在古典音乐方面的技术基础和对爵士乐的深刻理解为她赢得了大胆超越小提琴所能…

华为ICT七力助推文化产业新质生产力发展

创新起主导作用的新质生产力由新劳动者、新劳动对象、新劳动工具、新基础设施等四大要素共同构成&#xff0c;符合新发展理念的先进生产力质态&#xff1b;具有高科技、高能效、高质量等三大突出特征。而通过壮大新产业、打造新模式、激发新动能&#xff0c;新质生产力能够摆脱…

求m和n的最大公约数(C语言)

一、运行结果&#xff1b; 二、源代码&#xff1b; # define _CRT_SECURE_NO_WARNINGS # include <stdio.h>int main() {//初始化变量值&#xff1b;int remainder 1;int m 0;int n 0;int middle 0;//提示用户&#xff1b;printf("请输入整数m和n的值&#xff…