广告
返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >C++详细实现红黑树流程详解
  • 126
分享到

C++详细实现红黑树流程详解

2024-04-02 19:04:59 126人浏览 八月长安
摘要

目录红黑树的概念红黑树的性质红黑树的定义与树结构插入新增结点插入后维护红黑树性质的主逻辑旋转验证红黑树与AVl树的比较红黑树的应用红黑树的概念 红黑树,是一种二叉搜索树,但在每个结点

红黑树的概念

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

概念总结

红黑树是二叉搜索树的升级,结点里面存放的成员col标记当前结点的颜色,它的最长路径最多是最短路径的二倍,红黑树通过各个结点着色方式的限制接近平衡二叉树,但是不同于AVL的是AVL是一颗高度平衡的二叉树,红黑树只是接近平衡

红黑树的性质

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

红黑树性质总结:

1、红黑树结点的颜色只能是红色或者黑色

2、红黑树根节点必须是黑色

3、红黑树并没有连续的红色结点

4、红黑树中从根到叶子的每一条路径都包含相同的黑色结点

5、叶子是黑色,表示空的位置

最长路径和最短路径概念:

最短路径:从根结点到叶子结点每一条路径的结点颜色都是黑色的不包含红色

最长路径:红黑交替,黑色结点和红色结点的个数相等

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

假设结点个数为N,那么最短路径就是logN,最长路径就是2 * logN,所有并不存在最长路径超过最短路径2倍的情况

红黑树的定义与树结构

//枚举红黑颜色
enum colour 
{
	RED,
	BLACK,
};
//定义红黑树结点结构
template<class K,class V>
struct RBTreenode 
{
	//构造
	RBTreeNode(const pair<K, V>& kv = {0,0})
		:_left(nullptr)
		, _right(nullptr)
		, _parent(nullptr)
		, _kv(kv)
		,_col(BLACK)
	{ }
	//定义三叉链
	RBTreeNode<K, V>* _left; //左孩子
	RBTreeNode<K, V>* _right;//右孩子
	RBTreeNode<K, V>* _parent;  //父亲
	pair<K, V> _kv;  //pair对象
	//节点的颜色
	colour _col;  //定义枚举变量
};
//定义红黑树
template<class K, class V>
class RBTree 
{
		typedef RBTreeNode<K, V> Node;
	public:
		//构造
		RBTree() 
			:_root(nullptr)
		{}
	private:
		Node* _root;  //定义树的根节点
};

插入

插入过程类似搜索树的插入,重要的是维护红黑树的性质

pair<Node*, bool> Insert(const pair<K, V>& kv)
{
	if (!_root) //空树处理
	{
		_root = new Node(kv);
		_root->_col = BLACK;
		return { _root, true };
	}
	//二叉搜索树的插入逻辑
	Node* cur = _root, * parent = nullptr;
	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 { cur, false }; //插入失败
		}
	}
	cur = new Node(kv);
	cur->_col = RED;  //新增结点颜色默认设置为RED
	//判断插入结点是否在parent的左边或者右边
	if (parent->_kv.first > kv.first)  //左边
	{
		parent->_left = cur;
		cur->_parent = parent;
	}
	else     //右边	
	{
		parent->_right = cur;
		cur->_parent = parent;
	}
	
	while (parent && parent->_col == RED) //父亲存在且父亲为红色
	{
		Node* grandfather = parent->_parent;  //祖父
		//父亲出现在祖父的左边需要考虑的情况
		if(parent == grandfather ->left)
		{
			//1、uncle存在,uncle为红色
			
			//2、uncle不存在
			
				
			*/ 
		}
		else  //父亲出现在祖父的右边
		{
			Node* uncle = grandfather->_left; //叔叔在左子树 
			
			
			*/	
		}
	}
	//如果父亲不存在为了保证根结点是黑色的,这里一定得将根结点处理为黑色
	_root->_col = BLACK;
}

新增结点插入后维护红黑树性质的主逻辑

//1、父亲一定存在的情况,叔叔存在/不存在 父亲叔叔结点颜色为红色
while (parent && parent->_col == RED) //父亲存在且父亲为红色
{
	Node* grandfather = parent->_parent;  //祖父
	//如果父亲和叔叔结点颜色都是红色
	if (parent == grandfather->_left)  
	{
		Node* uncle = grandfather->_right;  
		if (uncle && uncle->_col == RED)  //对应情况:uncle存在且为红
		{
			//处理:父亲和叔叔变成黑色,祖父变成红色,继续向上调整
			uncle->_col = parent->_col = BLACK; 
			grandfather->_col = RED;
			//向上调整
			cur = grandfather;  //调整孩子
			parent = cur->_parent;//调整父亲
		}
		else   //uncle不存在,uncle存在且为黑
		{
			//直线情况(cur在parent的左边):只考虑单旋,以grandfather为旋转点进行右单旋转,
			//旋转完后将祖父的颜色变成红色,将父亲的颜色变成黑色
			if (parent->_left == cur) 
			{
				RotateR(grandfather);
				parent->_col = BLACK;
				grandfather->_col = RED;
			}
			else  //parent->_right == cur 
			{	
				//折线情况(cur在parent的右边):这里会引发双旋
				RotateL(parent);  //以parent为旋转点进行左单旋
				RotateR(grandfather); //以grandfather为旋转点进行右单旋转
				//旋转完后cur会去做树的根,那么设置为黑色,
				//为了保证每条路径的黑色结点个数相同,grandfather结点颜色设置为红
				cur->_col = BLACK;
				grandfather->_col = RED;  //黑色	结点个数相同
			}
		}
	}
	else //父亲在右子树
	{
			Node* uncle = grandfather->_left; //叔叔在左子树 
			if (uncle&& uncle->_col == RED)  //情况一处理:叔叔存在,且叔叔的颜色是红色的(包含了父亲的颜色是红色的情况)
			{
				//根据情况一处理即可:叔叔和父亲变黑,
				//祖父变红(目的是为了每条路径的黑色结点个数相同),继续向上
				cur = grandfather;  //孩子
				parent = cur->_parent;//父亲
			}
			else //叔叔不存在 
			{
				if (cur == parent->_right)  //新增结点在父亲的右边,直线情况左单旋处理
				{
					//左单旋转,以grandfather为旋转点,旋转完后parent去做新的根,grandfather去做左子树
					RotateL(grandfather);
					//调节颜色
					grandfather->_col = RED;
					parent->_col = BLACK;
				}
				else //新增结点在父亲的左边,折线情况,引发双旋
				{
					//处理:以parenrt为旋转点做右单旋,再以grandfather为旋转点做左单旋
					RotateR(parent);  //右旋
					RotateL(grandfather); //左旋
					parent->_col = grandfather->_col = RED;
					cur->_col = BLACK;
				}
				break;
			}
		}
	_root->_col = BLACK;
}

拆解讨论:

以下只列举parent在grandfather左边的情况,而parent在grandfather右边的情况处理方式只是反过来的,读者可以自行画图,这里仅留参考代码

Node* uncle = grandfather->_right;  
if (uncle && uncle->_col == RED)  //对应情况:uncle存在且为红
{
	//处理:父亲和叔叔变成黑色,祖父变成红色,继续向上调整
	uncle->_col = parent->_col = BLACK; 
	grandfather->_col = RED;
	//向上调整
	cur = grandfather;  //调整孩子
	parent = cur->_parent;//调整父亲
}

else   //uncle不存在,uncle存在且为黑
{
	//直线情况(cur在parent的左边):只考虑单旋,以grandfather为旋转点进行右单旋转,
	//旋转完后将祖父的颜色变成红色,将父亲的颜色变成黑色
	if (parent->_left == cur) 
	{
		RotateR(grandfather);
		parent->_col = BLACK;
		grandfather->_col = RED;
	}
	else  //parent->_right == cur 
	{	
		//双旋转
	}
}

//折线情况(cur在parent的右边):这里会引发双旋
RotateL(parent);  //以parent为旋转点进行左单旋
RotateR(grandfather); //以grandfather为旋转点进行右单旋转
//旋转完后cur会去做树的根,那么设置为黑色,
//为了保证每条路径的黑色结点个数相同,grandfather结点颜色设置为红
cur->_col = BLACK;
grandfather->_col = RED; 

旋转

void RotateR(Node* parent) //右单旋
	{
		Node* subL = parent->_left;
		Node* subLR = subL->_right;
		parent->_left = subLR;
		if (subLR) subLR->_parent = parent;  //防止subLR为nullptr
		subL->_right = parent;
		Node* parent_parent = parent->_parent; //指针备份
		parent->_parent = subL;
		if (_root == parent) //如果parent就是树的根 
		{
			_root = subL;  //subL取代parent
			_root->_parent = nullptr;
		}
		else  //如果parent并不是树的根
		{
			if (parent_parent->_left == parent) parent->_left = subL;
			else parent_parent->_right = subL;
			subL->_parent = parent_parent; //subL去做parent_parent的孩子
		}
	}
	//左单旋
	void RotateL(Node* parent)
	{
		Node* subR = parent->_right;
		Node* subRL = subR->_left;
		parent->_right = subRL;
		if (subRL) subRL->_parent = parent;
		subR->_left = parent;
		Node* parent_parent = parent->_parent;
		parent->_parent = subR;
		if (_root == parent)
		{
			_root = subR;
			_root->_parent = nullptr;
		}
		else
		{
			if (parent_parent->_left == parent) parent_parent->_left = subR;
			else parent_parent->_right = subR;
			subR->_parent = parent_parent;
		}
	}

验证


bool _CheckBlance(Node* root, int isBlackNum, int count) 
{
	if (!root) 
	{
		if (isBlackNum != count) 
		{
			printf("黑色结点个数不均等\n");
			return false;
		}
		return true; //遍历完整棵树,如果以上列举的非法情况都不存在就返回true
	}
	//检查是否出现连续的红色结点
	if (root->_col == RED && root->_parent->_col == RED) 
	{
		printf("出现了连续的红色结点\n");
		return false;
	} 
	//走前序遍历的过程中记录每一条路径黑色结点的个数
	if (root->_col == BLACK) count++;
	//递归左右子树
	return _CheckBlance(root->_left, isBlackNum, count) && 
			_CheckBlance(root->_right, isBlackNum, count);
}
//验证红黑树
bool CheckBlance()
{
	if (!_root) return true;  //树为null
	//根结点是黑色的
	if (_root->_col != BLACK) 
	{
		printf("根结点不是黑色的\n");
		return false;
	}
	//每一条路径黑色结点的个数必须是相同的,
	int isBlcakNum = 0;
	Node* left = _root; 
	while (left) 
	{
		if (left->_col == BLACK) isBlcakNum++; // 统计某一条路径的所以黑色结点个数
		left = left->_left;
	}
	//检查连续的红色结点,检查每一条路径的黑色结点个数是否相等
	return _CheckBlance(_root, isBlcakNum ,0);
}

红黑树与AVl树的比较

红黑树与AVL树的比较

红黑树和AVL树都是高效的平衡二叉树,增删改查的

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

红黑树的应用

  • c++ STL库 – map/set、mutil_map/mutil_set
  • Java 库
  • linux内核
  • 其他一些库

完整代码博主已经放在git上了,读者可以参考

红黑树实现.

到此这篇关于C++详细实现红黑树流程详解的文章就介绍到这了,更多相关C++红黑树内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: C++详细实现红黑树流程详解

本文链接: https://www.lsjlt.com/news/151082.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • C++详细实现红黑树流程详解
    目录红黑树的概念红黑树的性质红黑树的定义与树结构插入新增结点插入后维护红黑树性质的主逻辑旋转验证红黑树与AVl树的比较红黑树的应用红黑树的概念 红黑树,是一种二叉搜索树,但在每个结点...
    99+
    2022-11-13
  • C++超详细分析红黑树
    目录红黑树红黑树的概念红黑树的性质红黑树结点的定义红黑树的插入操作情况一情况二情况三红黑树的验证用红黑树封装map、set红黑树的迭代器封装map封装set红黑树 红黑树的概念 红黑...
    99+
    2022-11-13
  • C语言实现红黑树详细步骤+代码
    目录红黑树的概念红黑树的性质红黑树的定义与树结构插入新增结点插入后维护红黑树性质的主逻辑拆解讨论:旋转验证红黑树与AVl树的比较红黑树的应用总结红黑树的概念 红黑树,是一种二叉搜索树...
    99+
    2022-11-12
  • Java实现红黑树(平衡二叉树)的详细过程
    目录前言红黑二叉查找树2-3树2-3树的插入操作实现红黑二叉树结尾前言 在实现红黑树之前,我们先来了解一下符号表。 符号表的描述借鉴了Algorithms第四版,详情在:https...
    99+
    2022-11-12
  • C++ STL容器详解之红黑树部分模拟实现
    目录一、红黑树的概念二、红黑树的性质三、红黑树节点的定义四、红黑树结构 五、 红黑树的插入操作六、代码总结一、红黑树的概念 红黑树(Red Black Tree),是在计算机科学中用...
    99+
    2022-11-12
  • C++ 二叉树的实现超详细解析
    目录1、树的概念及结构(了解)1.1树的概念:1.2树的表示法:2、二叉树的概念及结构2.1二叉树的概念:2.2特殊的二叉树:2.3二叉树的性质:2.4二叉树的顺序存储:2.5二叉树...
    99+
    2022-11-13
  • C++超详细实现二叉树的遍历
    目录二叉树的遍历前序遍历中序遍历后序遍历层序遍历二叉树的遍历 Q:什么是二叉树的遍历? A:二叉树的遍历是指从根结点出发,按照某种次序依次访问二叉树中所有结点,使得每个结点被访问一次...
    99+
    2022-11-13
  • C语言实现扫雷游戏详细流程
    目录前言头文件部分主函数部分(源文件)函数定义(原文件)1.菜单打印2.初始化棋盘3.打印棋盘4.生成地雷5.数雷并赋值给trueboard6.排查雷区7.展开8.判断输赢前言 嘿!...
    99+
    2022-11-13
  • C语言详细实现猜拳游戏流程
    目录一、游戏逻辑二、思维导图三、游戏过程四、代码分析1.设置随机数的方法2.设置计算机出拳的方法3.判断输赢的方法4.玩家猜拳五、完整代码一、游戏逻辑 1.打印选择菜单(1.play...
    99+
    2022-11-13
  • c# 理解csredis库实现分布式锁的详细流程
    声明: 这里首先使用的是csredis,地址是https://github.com/2881099/csredis 该库本身已经足够完善,这里我画蛇添足一下,为了方便自己的使用。 本...
    99+
    2022-11-13
  • C++模拟实现vector流程详解
    目录模拟vector总结模拟vector 我们可以通过模板实现类似vector的类。我们实现一个StrVecTemp类,其内部通过allocator开辟空间,存储的类型用T来表示,T...
    99+
    2022-11-13
    C++ vector容器 C++ vector
  • C++实现AVL树的示例详解
    目录AVL 树的概念AVL 树的实现节点的定义接口总览插入旋转AVL 树的概念 也许因为插入的值不够随机,也许因为经过某些插入或删除操作,二叉搜索树可能会失去平衡,甚至可能退化为单链...
    99+
    2023-03-03
    C++实现AVL树 C++ AVL树
  • C语言与C++动态通讯录超详细实现流程
    目录1、思路以及要实现的功能2、详细步骤2.1 打印菜单界面(建一个源文件test.c)2.2 主函数2.3 初始化函数与加载函数2.4 增加联系人函数AddContact2.5 删...
    99+
    2022-11-13
  • C语言扫雷详细代码分步实现流程
    目录一,创建菜单二,创建游戏内容1.场景创建和初始化2.场景打印3.埋雷4.排雷完整代码1.game.h2.game.c3.test.c还是说一下:发的这些小游戏都是第一个版本,之后...
    99+
    2022-11-13
  • C++内存泄漏的检测与实现详细流程
    目录内存泄漏带来的问题难点hook实现泄漏判断与追踪(malloc和free重载)宏定义实现hook内存泄漏 malloc/new 调用在堆上分配的内存却没有相应的free/dele...
    99+
    2022-11-13
    C++ 内存泄漏检测 C++ 内存泄漏实现
  • C++ 中国象棋的实现流程详解
    中国象棋的中国棋文化,也是中华民族的文化瑰宝,它源远流长,趣味浓厚,基本规则简明易懂。中国象棋在中国的群众中基础远远超过围棋,是普及最广的棋类项目,中国象棋已流传到十几个国家和地区...
    99+
    2022-11-12
  • 线段树详解以及C++实现代码
    目录应用场景算法思想查询操作修改操作算法实现建树查询修改总结应用场景 假设有这样的问题:有n个数,m次操作,操作分为:修改某一个数或者查询一段区间的值 分析下,如果针对数组元素的修改...
    99+
    2022-11-12
  • c++详细讲解构造函数的拷贝流程
    #include <iostream> #include <string> using namespace std; void func(string str...
    99+
    2022-11-13
  • Vue渲染器设计实现流程详细讲解
    目录渲染器+响应系统渲染器基本原理DIY 渲染器渲染器+响应系统 最简渲染函数 使用以下函数渲染静态字符串或者动态拼接内容 // 渲染函数 function renderer(dom...
    99+
    2023-01-03
    Vue渲染器 Vue渲染器设计
  • C++基于boostasio实现synctcpserver通信流程详解
    目录一.功能介绍二.string类型数据交互2.1 程序源码2.2 编译&&执行2.3 程序执行结果三.byte类型数据交互3.1 程序源码3.2 编译&&a...
    99+
    2022-11-13
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作