C++语法(20)---- 模拟红黑树

这篇具有很好参考价值的文章主要介绍了C++语法(20)---- 模拟红黑树。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

C++语法(19)---- 模拟AVL树_哈里沃克的博客-CSDN博客https://blog.csdn.net/m0_63488627/article/details/130229501?spm=1001.2014.3001.5501

目录

1.红黑树介绍

2.模拟实现

1.枚举红黑颜色

2.节点的定义

3.树类框架

4.插入

5.检查

3.代码实现


1.红黑树介绍

C++语法(20)---- 模拟红黑树

最长路径不超过最短路径的两倍,近似平衡

最短:全黑

最长:一半黑一半红

1.每个节点不是红色就是黑色

2.根节点是黑色

3.如果一个节点是红色,它的两个孩子节点是黑色

4.每条路径都有相同数量的黑色节点

5.叶子节点(NIL节点)是黑的

满二叉树:全黑,或者每条路径一黑一红

较优情况:越均衡即越平衡

最差红黑树:左边全黑,右边一黑一红

较差情况:越不均衡

对比起AVL树,其实红黑树没有那么的较劲平衡,AVL的平衡得益于它不断的旋转。但是红黑树为了一些性能牺牲了平衡,减少了旋转的情况。

2.模拟实现

1.枚举红黑颜色

enum Color
{
	Black,
	Red,
};

2.节点的定义

template<class K, class V>
struct BRTreeNode
{
	pair<K, V> _kv;
	BRTreeNode* _left;
	BRTreeNode* _right;
	BRTreeNode* _parent;
	Color _col;

	BRTreeNode(const pair<K, V>& kv)
		:_kv(kv)
		, _left(nullptr)
		, _right(nullptr)
		, _parent(nullptr)
		, _col(Red)
	{}
};

定义一个节点时要注意其颜色:不过黑红的地位有所不同,选择决定后续的执行是否简单

选择红色:违背红色不能连续出现

选择黑色:违背整体路径黑色数量一致

我们认为选择红色更好,因为红色调节节点即可,黑色调节整个路径

3.树类框架

template<class K, class V>
class BRTree
{
private:
	Node* _root = nullptr;
};

4.插入

1.插入的头节点为根节点,根节点的颜色为黑色

2.普通插入的逻辑和搜索二叉树的逻辑一致

3.到这里就需判断是否两个红色节点连续,下面的问题就是调节双红色问题

调节双红色问题

情况一

C++语法(20)---- 模拟红黑树

 插入红色节点其父节点和叔叔节点都是红色,那么我们要调整父节点和叔叔节点为黑色,不过如果只是调整这一步,那么这条树的分支就比别的树黑色节点多一个,所以我们还要更新祖父节点为红色,但是祖父为红不能保证它的父节点是黑的,所以我们仍然要往上判断

C++语法(20)---- 模拟红黑树

情况二

单旋加变色

C++语法(20)---- 模拟红黑树

叔叔节点是黑色或者不存在,如果只是把父节点变黑是不够的,因为高度超了,所以我们要用到左右旋。 

情况三

双旋加变色

C++语法(20)---- 模拟红黑树

这样就变成了情况二 

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->_left;
		}
		else if (cur->_kv.first < kv.first)
		{
			parent = cur;
			cur = cur->_right;
		}
		else
			return false;
	}

	//走到这cur就是要插入的位置
	//cur要连接parent,parent也要连接cur---判断靠kv的大小
	cur = new Node(kv);
	if (parent->_kv.first > cur->_kv.first)
	{
		parent->_left = cur;
		cur->_parent = parent;
	}
	else
	{
		parent->_right = cur;
		cur->_parent = parent;
	}

	while (parent && parent->_col == Red)
	{
		Node* grandparent = parent->_parent;
		//parent分在grandparent左右
		if (grandparent->_left == parent)
		{
			//关键是看uncle节点不存在/红色/黑色的情况
			Node* uncle = grandparent->_right;
			//1.uncle红
			//parent和uncle变黑,grandparent变红
			//grandparent变红需要往上判断
			if (uncle && uncle->_col == Red)
			{
				grandparent->_col = Red;
				parent->_col = uncle->_col = Black;

				cur = grandparent;
				parent = cur->_parent;
			}
            else  //uncle不存在/黑色
			{
				//2.cur也是parent的左边,uncle不存在/黑色
				//右旋grandparents,parent变黑,
				if (cur == parent->_left)
				{
					_RotateR(grandparent);
					parent->_col = Black;
					grandparent->_col = Red;
				}
				//3.cur是parent的右边,uncle不存在/黑色
				//左旋parent再右旋grandparents,cur变黑,grandparents变红
				else
				{
					_RotateL(parent);
					_RotateR(grandparent);
					cur->_col = Black;
					grandparent->_col = Red;
				}

				//抽象树的头被设置为黑色,对上面没有影响,所以不需要进行循环
				break;
			}
				
		}
        else
		{
		Node* uncle = grandparent->_left;

		    //1.uncle红
		    //parent和uncle变黑,grandparent变红
		    //grandparent变红需要往上判断
		    if (uncle && uncle->_col == Red)
		    {
			    grandparent->_col = Red;
			    parent->_col = uncle->_col = Black;

			    cur = grandparent;
			    parent = cur->_parent;
		    }
			else  //uncle不存在/黑色
			{
				//2.cur也是parent的右边,uncle不存在/黑色
				//左旋grandparents,parent变黑,
				if (cur == parent->_right)
				{
					_RotateL(grandparent);
					parent->_col = Black;
					grandparent->_col = Red;
				}
				//3.cur是parent的右边,uncle不存在/黑色
				//右旋parent再左旋grandparents,cur变黑,grandparents变红
				else
				{
					_RotateR(parent);
					_RotateL(grandparent);
					cur->_col = Black;
					grandparent->_col = Red;
				}

				//抽象树的头被设置为黑色,对上面没有影响,所以不需要进行循环
				break;
			}
		}
	}
	_root->_col = Black;
	return true;
}

5.检查

inspect函数

1.空树返回true

2.根节点是否是红的

3.先遍历最左边,得到这个分支的黑色节点数作为参考

check函数

1.传入参考黑色节点个数

2.计数每一分支的黑色节点数,递归到底部,可对照节点数是否满足

3.递归遍历看看有没有连续的红色节点文章来源地址https://www.toymoban.com/news/detail-427249.html

private: 
bool check(Node* root, size_t& reference, size_t num)
{
	if (root == nullptr)
	{
		if (num != reference)
		{
			cout << "路径长度有问题" << endl;
			return false;
		}

		return true;
	}
			

	if (root->_col == Red && root->_parent && root->_parent->_col == Red)
	{
		cout << "节点连续红色" << endl;
		return false;
	}
	
	if (root->_col == Black)
		num++;

	return check(root->_left, reference, num) && check(root->_right, reference, num);
}

bool _Inspect(Node* root)
{
	//空树也是红黑树
	if (_root == nullptr)
		return true;

	//检测根节点是否为黑色
	if (_root->_col != Black)
	{
		cout << "根节点是红色的" << endl;
		return false;
	}
			
	size_t leftNum = 0;
	Node* cur = _root;
	while (cur)
	{
		if (cur->_col == Black)
				leftNum++;
			cur = cur->_left;
	}
	//检测所有路径黑色节点的数量是否一样
	//检测相邻节点是不是都是红色的
	return check(_root, leftNum, 0);
}

3.代码实现

#pragma once
#include<iostream>
#include<assert.h>
#include <stdlib.h>
#include<time.h>
using namespace std;

enum Color
{
	Black,
	Red,
};

template<class K, class V>
struct BRTreeNode
{
	pair<K, V> _kv;
	BRTreeNode* _left;
	BRTreeNode* _right;
	BRTreeNode* _parent;
	Color _col;

	BRTreeNode(const pair<K, V>& kv)
		:_kv(kv)
		, _left(nullptr)
		, _right(nullptr)
		, _parent(nullptr)
		, _col(Red)
	{}
};

template<class K, class V>
class BRTree
{
public:
	typedef BRTreeNode<K, V> Node;
	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->_left;
			}
			else if (cur->_kv.first < kv.first)
			{
				parent = cur;
				cur = cur->_right;
			}
			else
				return false;
		}

		//走到这cur就是要插入的位置
		//cur要连接parent,parent也要连接cur---判断靠kv的大小
		cur = new Node(kv);
		if (parent->_kv.first > cur->_kv.first)
		{
			parent->_left = cur;
			cur->_parent = parent;
		}
		else
		{
			parent->_right = cur;
			cur->_parent = parent;
		}

		while (parent && parent->_col == Red)
		{
			Node* grandparent = parent->_parent;
			//parent分在grandparent左右
			if (grandparent->_left == parent)
			{
				//关键是看uncle节点不存在/红色/黑色的情况
				Node* uncle = grandparent->_right;
				//1.uncle红
				//parent和uncle变黑,grandparent变红
				//grandparent变红需要往上判断
				if (uncle && uncle->_col == Red)
				{
					grandparent->_col = Red;
					parent->_col = uncle->_col = Black;

					cur = grandparent;
					parent = cur->_parent;
				}
				else  //uncle不存在/黑色
				{
					//2.cur也是parent的左边,uncle不存在/黑色
					//右旋grandparents,parent变黑,
					if (cur == parent->_left)
					{
						_RotateR(grandparent);
						parent->_col = Black;
						grandparent->_col = Red;
					}
					//3.cur是parent的右边,uncle不存在/黑色
					//左旋parent再右旋grandparents,cur变黑,grandparents变红
					else
					{
						_RotateL(parent);
						_RotateR(grandparent);
						cur->_col = Black;
						grandparent->_col = Red;
					}

					//抽象树的头被设置为黑色,对上面没有影响,所以不需要进行循环
					break;
				}
				
			}
			else
			{
				Node* uncle = grandparent->_left;

				//1.uncle红
				//parent和uncle变黑,grandparent变红
				//grandparent变红需要往上判断
				if (uncle && uncle->_col == Red)
				{
					grandparent->_col = Red;
					parent->_col = uncle->_col = Black;

					cur = grandparent;
					parent = cur->_parent;
				}
				else  //uncle不存在/黑色
				{
					//2.cur也是parent的右边,uncle不存在/黑色
					//左旋grandparents,parent变黑,
					if (cur == parent->_right)
					{
						_RotateL(grandparent);
						parent->_col = Black;
						grandparent->_col = Red;
					}
					//3.cur是parent的右边,uncle不存在/黑色
					//右旋parent再左旋grandparents,cur变黑,grandparents变红
					else
					{
						_RotateR(parent);
						_RotateL(grandparent);
						cur->_col = Black;
						grandparent->_col = Red;
					}

					//抽象树的头被设置为黑色,对上面没有影响,所以不需要进行循环
					break;
				}
			}
		}
		_root->_col = Black;
		return true;
	}

	void Print()
	{
		_Print(_root);
		cout << endl;
	}

	bool Inspect()
	{
		return _Inspect(_root);
	}

private: 
	bool check(Node* root, size_t& reference, size_t num)
	{
		if (root == nullptr)
		{
			if (num != reference)
			{
				cout << "路径长度有问题" << endl;
				return false;
			}

			return true;
		}
			

		if (root->_col == Red && root->_parent && root->_parent->_col == Red)
		{
			cout << "节点连续红色" << endl;
			return false;
		}
		
		if (root->_col == Black)
			num++;

		return check(root->_left, reference, num) && check(root->_right, reference, num);
	}

	bool _Inspect(Node* root)
	{
		//空树也是红黑树
		if (_root == nullptr)
			return true;

		//检测根节点是否为黑色
		if (_root->_col != Black)
		{
			cout << "根节点是红色的" << endl;
			return false;
		}
			
		size_t leftNum = 0;
		Node* cur = _root;
		while (cur)
		{
			if (cur->_col == Black)
				leftNum++;
			cur = cur->_left;
		}
		//检测所有路径黑色节点的数量是否一样
		//检测相邻节点是不是都是红色的
		return check(_root, leftNum, 0);
	}

	void _Print(Node*& cur)
	{
		if (cur == nullptr)
			return;
		_Print(cur->_left);
		cout << cur->_kv.first << " ";
		_Print(cur->_right);
	}

	void _RotateL(Node*& parent)
	{
		Node* pparent = parent->_parent;
		Node* SubR = parent->_right;
		Node* SubRL = SubR->_left;
		if (pparent == nullptr)
		{
			_root = SubR;
			SubR->_parent = nullptr;
		}
		else
		{
			if (pparent->_left == parent)
				pparent->_left = SubR;
			else
				pparent->_right = SubR;
			SubR->_parent = pparent;
		}
		parent->_parent = SubR;
		SubR->_left = parent;
		parent->_right = SubRL;
		if (SubRL != nullptr)
			SubRL->_parent = parent;
	}

	void _RotateR(Node*& parent)
	{
		Node* pparent = parent->_parent;
		Node* SubL = parent->_left;
		Node* SubLR = SubL->_right;
		if (pparent == nullptr)
		{
			_root = SubL;
			SubL->_parent = nullptr;
		}
		else
		{
			if (pparent->_left == parent)
				pparent->_left = SubL;
			else
				pparent->_right = SubL;
			SubL->_parent = pparent;
		}
		parent->_parent = SubL;
		SubL->_right = parent;
		parent->_left = SubLR;
		if (SubLR != nullptr)
			SubLR->_parent = parent;
	}

	Node* _root = nullptr;
};

到了这里,关于C++语法(20)---- 模拟红黑树的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【C++】详解红黑树并模拟实现

    前言:        上篇文章我们一起学习了AVL树比模拟实现,我们发现AVL树成功地把时间复杂度降低到了O(logN)。但是同时我们不难发现一个问题,在构建AVL树中我们也付出了不小的代价,频繁的旋转操作导致效率变低。为了解决这个问题,我们本章将迎来更为实用的红黑树,他

    2024年02月09日
    浏览(43)
  • 【C++】红黑树模拟实现插入功能(包含旋转和变色)

    本篇主要讲解红黑树的模拟实现,实现之后再封装成map和set。 红黑树中所用到的旋转功能均源自我的上一篇博客,本篇中不会再详谈旋转,若对于旋转不了解的同学可以先看看上一篇博客:【C++】AVL树模拟实现插入功能 前一篇的AVL树,是严格平衡的一棵二叉搜索树,也就是

    2024年02月13日
    浏览(53)
  • 【C++】 使用红黑树模拟实现STL中的map与set

    前面的文章我们学习了红黑树,也提到了C++STL中的map和set的底层其实就是用的红黑树来实现的(而map和set的使用我们前面也学过了)。 既然红黑树我们也学习过了,那这篇文章我们就用红黑树来简单实现一下STL中的map和set,重点是学习它的框架。 上一篇文章我们实现了红黑

    2024年02月12日
    浏览(31)
  • 【C++高阶(四)】红黑树深度剖析--手撕红黑树!

    💓博主CSDN主页:杭电码农-NEO💓   ⏩专栏分类:C++从入门到精通⏪   🚚代码仓库:NEO的学习日记🚚   🌹关注我🫵带你学习C++   🔝🔝 如果说发明AVL树的人是天才,那么 发明红黑树的人可以称为天才中的 精英!为什么AVL树这么强大但是没啥 人用呢?就是因为红黑树比你还好

    2024年02月05日
    浏览(46)
  • 【C++进阶5-红黑树】噩梦般的存在?手提AVLTree暴揍红黑树!

    今天,带来无数人的噩梦——红黑树的讲解。文中不足错漏之处望请斧正! 如果还没看过AVLTree讲解的一定要去看看,看完才能更好理解红黑树! 红黑树是自平衡的二叉搜索树。 红黑树的规则: 每个结点非黑即红 根结点为黑 叶子结点为黑(此处的叶子结点指空结点) 不能有

    2024年02月07日
    浏览(45)
  • 【C++】:红黑树

    朋友们、伙计们,我们又见面了,本期来给大家解读一下有关多态的知识点,如果看完之后对你有一定的启发,那么请留下你的三连,祝大家心想事成! C 语 言 专 栏: C语言:从入门到精通 数据结构专栏: 数据结构 个  人  主  页 : stackY、 C + + 专 栏   : C++ Linux 专 栏

    2024年02月04日
    浏览(39)
  • C++之红黑树

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

    2024年02月09日
    浏览(42)
  • C++ 红黑树

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

    2024年02月05日
    浏览(42)
  • 【C++】实现红黑树

    红黑树是一个二叉搜索树,与AVL树相比,红黑树不再使用平衡因子来控制树的左右子树高度差,而是用颜色来控制平衡,颜色为红色或者黑色。控制任意一条从根到叶子节点的路径的节点颜色,就可以确保没有一条路径会比其他路径长出俩倍,因而是接近平衡的。 红黑树的规

    2024年03月25日
    浏览(42)
  • C++红黑树

    前置说明: 需要学习AVL树的旋转之后再来看红黑树的旋转 因为红黑树的旋转是复用的AVL树的旋转的: 大家可以看我的这篇博客,里面详细介绍了AVL树的四种旋转 C++ AVL树(四种旋转,插入) 对于颜色我们采用枚举类型定义 对于红黑树节点,依旧采用Key-Value模型存储pair 1.共识 首先我们

    2024年02月04日
    浏览(40)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包