红黑树插入数据的底层详解

news2025/1/11 16:50:40

红黑树定义

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

红黑树节点实现

enum COLOR {
	RED,
	BLACK
};

template<class K,class V>
struct ListNode {				// 节点
	//typedef ListNode<K, V> node;
	using Node = ListNode<K, V>;
	ListNode(std::pair<K,V> kv):_kv(kv) {
		_parent = _left = _right = nullptr;
		_color = RED;
	}
	Node* _parent;
	Node* _left;
	Node* _right;
	std::pair<K, V> _kv;
	COLOR _color;
};

简单点来说就是一个结构体,里面三个指针,一个pair<k,v>数据,一个颜色(enum变量),三个指针是父亲,左孩子,右孩子

依照二叉搜索树的定义,我们来实现插入操作

1.接口

这里我按stl库里set,map的insert函数最主要的模型进行实现

2.找到待插入的位置

按搜索二叉树的要求,比根节点大的去右子树,比根节点小的去左子树,如果和根节点相等则插入失败,否则在子树继续比较,直到找到叶节点,

Node* new_node = new Node(kv);
_size++;
if (_root == nullptr) {
	new_node->_color = BLACK;
	_root = new_node;
	return std::make_pair(iterator(new_node), true);
}
Node* child = _root;
Node* father = _root;	// 父子交替,找 新节点应该插入的地方
while (child) {			//儿子不是空
	father = child;
	if (child->_kv.first > kv.first) {
		child = child->_left;
	}
	else if (kv.first > child->_kv.first) {
		child = child->_right;
	}
	else {
		
		_size--;
		return std::make_pair(iterator(child), false);
	}
}

3.插入新节点

		void add_node(Node* father, Node* new_node) {
			if (new_node->_kv.first > father->_kv.first) {
				father->_right = new_node;
			}
			else {
				father->_left = new_node;
			}
			new_node->_parent = father;
		}

插入完成,依照红黑树的定义对原树进行重构

把红黑树的定义拉过来

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

默认新节点的颜色为什么是红色

如果新节点是黑色,新插入节点所在路径上的颜色一定不同于其他路径.定义4失效,此时需要更改某些路径上的黑色节点,最差情况需要重构整颗树

如果新节点是红色,如果父亲节点是红色的话定义三失效,此时需要对所在子树进行选择,最差情况也不过是选择一条逻辑,如果插入的是根节点,也会违反定义5,此时只需直接修改颜色即可

对比之下,新节点为红色时,重构树的代价最小,

重构

情况一,没有父节点

此时插入的节点只可能是根节点,直接修改颜色

情况二,父亲节点为黑色

此时不与红黑树冲突,不需要重构

情况三,父亲节点为红色

父亲为红色,由于性质三的限制,爷爷节点不能为红色,必定为黑色

此时需要注意叔叔节点的颜色

情况三_一.没有叔叔节点  

这种情况不能单靠修改颜色来重构树,需要进行旋转

关于父子所在不同位置进行不同的旋转可以参考AVL树底层原理+模拟实现-CSDN博客

简单来说:

  1. 父亲为爷爷左,孩子为父亲左,则右旋父亲
  2. 父亲为爷爷右,孩子为父亲右,则左旋父亲
  3. 父亲为爷爷左,孩子为父亲右,则先左旋孩子,再右旋孩子
  4. 父亲为爷爷右,孩子为父亲左,则先右旋孩子,再左旋孩子

关于颜色

 如果父亲孩子同侧   父亲   ->   黑色   爷爷     ->   红色     如果父亲孩子异侧    新节点    ->   黑色     爷爷    ->   红色

			if (uncle == nullptr ) {							// 叔叔为空,需要发生旋转
				Node* tmp = revorve(new_node, parent, grandparent);
				if (tmp == _root) {
					tmp->_color = BLACK;
				}
				else {
					reconstitution(tmp);
				}
			}
情况三_二.叔叔节点为红色

这种情况可以靠修改颜色来完成重构,但是需要注意,此时爷爷节点被强制修改为了黑色,此时就需要再对爷爷节点进行重构

else if (uncle->_color == RED) {
	// 叔叔为红色 需要改变颜色,并继续向上重构,因为新的爷爷节点被强制改为红色,需要继续向上重构
	uncle->_color = parent->_color = BLACK;
	if (grandparent != _root) {
		// 爷爷节点不为root的话
		grandparent->_color = RED;
		reconstitution(grandparent);
	}
}
情况三_三叔叔节点为黑色

这种情况需要先将新节点与父节点旋转为同侧,再旋转新的父节点,与爷爷的颜色互换(这张图只是为了便于观看理解过程!)

if (uncle->_color == BLACK) {							// 叔叔为空,需要发生旋转
	Node* tmp = revorve(new_node, parent, grandparent);
	if (tmp == _root) {
		tmp->_color = BLACK;
	}
	else {
		reconstitution(tmp);
	}
}
附:为什么会出现情况三_三

依照红黑树的定义四:每条路径上的黑色元素数量相等,显然下图中的红黑树本身就已经不合定义了,那么为什么还会出现这种情况呢?

在情况三_三中如果我们假设被重构的节点是新加入的节点的话,显然是不符合定义的,但是,除了是新加入的节点以外,还有一种情况,就是情况三_二中继续向上重构的节点.

这种递归式的重构就会出现情况三_三了

总结

  1. 插入节点为根节点,进行重构                ------    修改根节点颜色为黑色
  2. 父亲为红色,没有叔叔节点  进行重构   -------   将父节点与子节点旋转至同侧,旋转父节点    。 如果父亲孩子同侧   父亲   ->   黑色   爷爷     ->   红色     如果父亲孩子异侧    新节点    ->   黑色     爷爷    ->   红色
  3. 父亲为红色,叔叔节点为红色             --------  父亲与叔叔节点修改为黑色,爷爷节点修改为红色,继续重构爷爷节点
  4. 父亲为红色,叔叔节点为黑色             -------- 将父节点与子节点旋转至同侧,旋转父节点。 如果父亲孩子同侧   父亲   ->   黑色   爷爷     ->   红色     如果父亲孩子异侧    新节点    ->   黑色     爷爷    ->   红色

模拟实现参考代码

		std::pair<iterator,bool> insert(const std::pair<K, V>& kv) {
			Node* new_node = new Node(kv);
			_size++;
			if (_root == nullptr) {
				new_node->_color = BLACK;
				_root = new_node;
				return std::make_pair(iterator(new_node), true);
			}
			Node* child = _root;
			Node* father = _root;	// 父子交替,找 新节点应该插入的地方
			while (child) {			//儿子不是空
				father = child;
				if (child->_kv.first > kv.first) {
					child = child->_left;
				}
				else if (kv.first > child->_kv.first) {
					child = child->_right;
				}
				else {
					
					_size--;
					return std::make_pair(iterator(child), false);
				}
			}
			//此处儿子是空,代表新节点应该插入的地方
			add_node(father, new_node);
			//对插入的节点进行检查,不合理的地方重构
			reconstitution(new_node);
			return std::make_pair(iterator(new_node), true);

		}
	protected:
		void add_node(Node* father, Node* new_node) {
			if (new_node->_kv.first > father->_kv.first) {
				father->_right = new_node;
			}
			else {
				father->_left = new_node;
			}
			new_node->_parent = father;
		}
		void reconstitution(Node* new_node) {
			/*
			1. 每个结点不是红色就是黑色 
			2. 根节点是黑色的  
			3. 如果一个节点是红色的,则它的两个孩子结点是黑色的  
			4. 对于每个结点,从该结点到其所有后代叶结点的简单路径上,均 包含相同数目的黑色结点  
			5. 每个叶子结点都是黑色的(此处的叶子结点指的是空结点)
			*/
			if (new_node == _root) {
				return;
			}
			if (new_node->_parent->_color == BLACK)	
				// 父亲是黑色,插入红色节点,符合红黑树要求,不做处理
				return;
			// 父亲是红色,此时不符合条件三,需要发生重构
			Node* parent = new_node->_parent;				// 新节点的父节点
			Node* grandparent = new_node->_parent->_parent; // 爷爷节点
			Node* uncle = nullptr;							// 叔叔节点
			if (grandparent->_left == parent) {				// 叔叔节点需要通过父亲节点与爷爷节点的关系来判断
				uncle = grandparent->_right;			
			}
			else {
				uncle = grandparent->_left;
			}
			

			// 判断情况
			if (uncle == nullptr || uncle->_color == BLACK) {							// 叔叔为空,需要发生旋转
				Node* tmp = revorve(new_node, parent, grandparent);
				if (tmp == _root) {
					tmp->_color = BLACK;
				}
				else {
					reconstitution(tmp);
				}
			}
			else if (uncle->_color == RED) {
				// 叔叔为红色 需要改变颜色,并继续向上重构,因为新的爷爷节点被强制改为红色,需要继续向上重构
				uncle->_color = parent->_color = BLACK;
				if (grandparent != _root) {
					// 爷爷节点不为root的话
					grandparent->_color = RED;
					reconstitution(grandparent);
				}
			}
			//else {	// 叔叔节点为黑色
			//	// 旋转
			//	reconstitution(revorve(new_node, parent, grandparent));
			//}
		}
		Node* revorve(Node* node,Node* parent,Node* grandparnt) {
			if (node == parent->_left && parent == grandparnt->_left) {
				// 子为父左,父为爷左,右旋父
				revorveR(parent);
				node->_color = BLACK;
				return parent;
			}
			else if(node == parent->_left && parent == grandparnt->_right) {
				// 子为父左,父为爷右,右旋子,左旋子
				revorveR(node);
				revorveL(node);
				parent->_color = BLACK;
				return node;
			}
			else if (node == parent->_right && parent == grandparnt->_right) {
				// 子为父右,父为爷右,左旋父
				revorveL(parent);
				node->_color = BLACK;
				return parent;

			}
			else {
				// 子为父右,父为爷左,左旋子,右旋子
				revorveL(node);
				revorveR(node);
				parent->_color = BLACK;
				return node;

			}
		}
		void revorveL(Node* node)
		{
			// 左旋需要改变node节点的 1.本身 2.父节点 3.爷爷节点 4.左孩子节点 共计四个节点
			// 记录四个节点的值,防止等下修改时混乱
			Node* parent = node->_parent;
			Node* grandparent = parent->_parent;
			Node* leftchild = node->_left;
			// 1.修改本身节点
			node->_parent = grandparent;
			node->_left = parent;

			// 2.修改父节点
			parent->_parent = node;
			parent->_right = leftchild;
			if (parent == _root)
				_root = node;

			// 3. 修改爷爷节点
			if (grandparent != nullptr) {
				if (grandparent->_left == parent)
					grandparent->_left = node;
				else {
					grandparent->_right = node;
				}
			}

			// 4. 修改左孩子节点
			if(leftchild != nullptr)
				leftchild->_parent = parent;
		}
		void revorveR(Node* node)
		{
			// 右旋需要改变node节点的 1.本身 2.父节点 3.爷爷节点 4.右孩子节点 共计四个节点
			// 记录四个节点的值,防止等下修改时混乱
			Node* parent = node->_parent;
			Node* grandparent = parent->_parent;
			Node* rightchild = node->_right;
			// 1.修改本身节点
			node->_parent = grandparent;
			node->_right = parent;

			// 2.修改父节点
			parent->_parent = node;
			parent->_left = rightchild;

			// 3. 修改爷爷节点
			if (grandparent != nullptr) {
				if (grandparent->_left == parent)
					grandparent->_left = node;
				else {
					grandparent->_right = node;
				}
			}

			// 4. 修改右孩子节点
			if(rightchild!=nullptr)
				rightchild->_parent = parent;
		}

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

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

相关文章

c语言---循环 、判断基础知识详解

if语句 else离最近的if语句结合。 if语句题目 //1. 判断一个数是否为奇数 //2. 输出1 - 100之间的奇数 #include <stdio.h> int main() {int n 0;scanf("%d", &n);if (n % 2){printf("奇数\n");}else{printf("不是奇数\n"…

LeetCode 2813.子序列最大优雅度

给你一个长度为 n 的二维整数数组 items 和一个整数 k 。 items[i] [profiti, categoryi]&#xff0c;其中 profiti 和 categoryi 分别表示第 i 个项目的利润和类别。 现定义 items 的 子序列 的 优雅度 可以用 total_profit distinct_categories^2 计算&#xff0c;其中 t…

VScode如何调试

调试 1.打断点 1.点击调试按钮 3.点击下拉选择环境node&#xff0c;点击绿三角选择输入调试的命令&#xff08;具体命令查看package.json中scripts中的哪一个命令和运行的文件&#xff09;&#xff0c;点击右边的设置&#xff08;可以直接跳下面第八步&#xff01;&#xff…

【2024最新精简版】SpringCloud面试篇

文章目录 SpringBoot和SpringCloud什么区别 ?你们项目为什么要使用微服务Spring Cloud 5大组件有哪些&#xff1f;&#x1f44d;什么是微服务?微服务的优缺点是什么?你们项目中微服务之间是如何通讯的? &#x1f44d;服务注册和发现是什么意思&#xff1f;Spring Cloud 如何…

LeetCode题练习与总结:被围绕的区域--130

一、题目描述 给你一个 m x n 的矩阵 board &#xff0c;由若干字符 X 和 O 组成&#xff0c;捕获 所有 被围绕的区域&#xff1a; 连接&#xff1a;一个单元格与水平或垂直方向上相邻的单元格连接。区域&#xff1a;连接所有 0 的单元格来形成一个区域。围绕&#xff1a;如果…

使用pytest-xdist实现分布式APP自动化测试

不知道大家有没有遇到这样一种情况&#xff0c;实际工作中&#xff0c;app自动化测试的用例可能是成百上千条的&#xff0c;如果放在一台机器上跑&#xff0c;消耗的时间非常久&#xff0c;那能不能使用分布式的来跑测试用例呢&#xff1f;比如有1000条测试用例&#xff0c;给A…

骨传导耳机品牌排行前五名揭晓:精选5款音质卓越、佩戴舒适的优选产品!

骨传导耳机是目前非常热门的蓝牙耳机&#xff0c;有很多人都想去尝试&#xff0c;但又很多消费者再入手后&#xff0c;都出现了佩戴不舒服&#xff0c;音质刺耳等问题&#xff0c;作为一位拥有十多年经验的数码测评师&#xff0c;我有必要提醒大家&#xff0c;尽管市面上各种骨…

inpaint下载安装2024-inpaint软件安装包下载v5.0.6官网最新版附加详细安装步骤

Inpaint软件最新版是一款功能强大的图片去水印软件&#xff0c;这款软件拥有强大的智能算法&#xff0c;能够根据照片的背景为用户去除照片中的各种水印&#xff0c;并修补好去除水印后的图片。并且软件操作简单、界面清爽&#xff0c;即使是修图新手也能够轻松上手&#xff0c…

什么牌子的灯好不伤眼?带你了解什么灯对眼睛伤害最小

眼睛是人类获取信息最重要的感官器官之一&#xff0c;而近视则会导致视力模糊&#xff0c;进而影响学习效果和生活品质。因此&#xff0c;什么灯对眼睛伤害最小成为许多人迫切寻找的目标。本文将为各位家长解答目前许多家长选择为孩子保护视力的产品——护眼台灯。护眼台灯以其…

如何打开mobi文件?两个步骤解决

打开MOBI格式的电子书&#xff0c;其实相当简便。NeatReader作为一个兼容多格式多系统的电子书阅读器&#xff0c;对MOBI格式的支持自然不在话下。下面是使用NeatReader阅读MOBI文件的步骤&#xff1a; 第一步&#xff1a;下载并安装NeatReader&#xff1a; 首先&#xff0c;你…

如何利用Python处理站点数据、格点观测数据、再分析ERA5;GLDAS、遥感数据、水文数据、气象数据、陆面模式数据、气候变化数据等

Python是功能强大、免费、开源&#xff0c;实现面向对象的编程语言&#xff0c;Python能够运行在Linux、Windows、Macintosh、AIX操作系统上及不同平台&#xff08;x86和arm&#xff09;&#xff0c;Python简洁的语法和对动态输入的支持&#xff0c;再加上解释性语言的本质&…

ECharts 蓝色系-荧光图标折线图01案例

ECharts 蓝色系-荧光图标折线图01案例 图表意义 本折线图案例展示了一周内不同路线的使用情况或数据统计。通过折线的上升和下降&#xff0c;可以直观地观察到每条路线的流量或数据变化趋势&#xff0c;从而进行分析和决策。 效果预览 效果图展示不同路线的数据统计和个性化…

下载依赖有问题(只有自己有问题)

有缓存&#xff01; 删除node_modules 命令&#xff1a;npm run clean 前提是该项目支持这个命令&#xff1a;package.json > scripts 内有 clean 例如下面这个就没有clean&#xff0c;则直接手动删除 清除缓存 npm cache clean --force pnpm store prune删除lock文件 …

Linux——ansible剧本

剧本&#xff08;playbook&#xff09; 现在&#xff0c;可以写各种临时命令 但如果&#xff0c;想把所有步骤&#xff0c;集合到一起&#xff0c;写到同一个文件里 让ansible自动按顺序执行 就必须要写“剧本” 剧本里面&#xff0c;也可以写临时命令&#xff0c;但是剧本…

AI辅助写作:如何高效完成毕业论文

写作这件事一直让我们从小学时期就开始头痛&#xff0c;初高中时期800字的作文让我们焦头烂额&#xff0c;一篇作文里用尽了口水话&#xff0c;拼拼凑凑才勉强完成。 大学时期以为可以轻松顺利毕业&#xff0c;结果毕业前的最后一道坎拦住我们的是毕业论文&#xff0c;这玩意不…

Nature发文介绍使用ChatGPT帮助学术写作的三种方式

文章链接&#xff1a;https://www.nature.com/articles/d41586-024-01042-3 一、介绍 这篇文章是由Dritjon Gruda撰写的&#xff0c;讨论了生成性人工智能&#xff08;AI&#xff09;在学术写作、编辑和同行评审中的三种应用方式。Gruda认为&#xff0c;尽管学术界对聊天机器…

opengauss安装postgis插件(Docker部署)

opengauss安装postgis插件 当然不管是安装opengauss还是给其安装插件,对其官方文档的解读是至关重要的,opengauss官网 点击最新开发版本进入快速入门链接。则可查看具体的各种指南。本次我使用的是极简版-容器安装。 下载源码并修改版本号 从官网的配置准备中可以发现,我们…

?? 与 || 在 JavaScript 中的微妙差别

起初&#xff0c;你可能会认为你可以随意替换任何你喜欢的人&#xff0c;对吗&#xff1f; 错误。他们并非你所想的那样。 我们必须一劳永逸地学习这个区别&#xff0c;以避免日后出现痛苦的错误。 这个差别是什么&#xff1f; 这是他们对待真值和假值的令人难以置信的对比。这…

这个是 2024 Idea最新激活码

idea的激活与安装 操作如下&#xff1a; ① 打开网站&#xff1a;https://web.52shizhan.cn 切换到&#xff1a;正版激活码&#xff0c;点击获取 ② 获取后的激活码&#xff0c;到idea里打开help->register 打开弹窗&#xff0c;如图 切换的activate code 输入激活码&…

Conmi的正确答案——Vue默认加载方式设置为Yarn后怎么修改

Vue版本&#xff1a;3 1和2主要是搜索文件所在位置&#xff0c;Windows的这个文件一般在“C:\User{当前用户}”下&#xff0c;linux的非root情况下一般在“/home/{当前用户}”下。 1、打开“Everything”&#xff1b; 2、搜索“vuerc”&#xff1b; 3、打开“.vuerc”&#xf…