【ONE·C++ || set和map(二):AVL树和红黑树】

news2025/1/23 4:12:08

总言

  主要介绍AVL树和红黑树的部分框架结构和特性

文章目录

  • 总言
  • 1、AVL树
    • 1.1、基本概念介绍
    • 1.2、重要框架与特性实现
      • 1.2.1、如何搭建AVL树的结点 :AVLTreeNode,引入三叉链
      • 1.2.2、如何搭建AVL树
      • 1.2.3、AVL树插入讲解
    • 1.3、AVL树插入
      • 1.3.1、step1:按照二叉搜索树的方式插入新节点
      • 1.3.2、step2:调整节点的平衡因子
        • 1.3.2.1、更新平衡因子的相关规则
        • 1.3.2.2、AVL树旋转模式(四类)
          • 1.3.2.2.1、左单旋:新结点插入较高右子树的右侧(右右)
          • 1.3.2.2.2、右单旋:新结点插入较高左子树的左侧(左左)
          • 1.3.2.2.3、左右双旋:新结点插入较高左子树的右侧(左右)
          • 1.3.2.2.4、右左双旋:新结点插入较高右子树的左侧(右左)
      • 1.3.3、整体实现总览与遍历检查
          • 1.3.3.1、遍历检查
          • 1.3.3.2、整体展示
  • 2、红黑树
    • 2.1、基本概念与性质
    • 2.2、重要框架与特性实现
      • 2.2.1、如何搭建红黑树的结点
      • 2.2.2、如何搭建红黑树
      • 2.2.3、红黑树的插入讲解
    • 2.3、红黑树的插入
      • 2.3.1、step1:按照二叉搜索树的方式插入新节点
      • 2.3.2、step2:检查新增结点后红黑树性质是否遭到破坏
      • 2.3.3、整体实现总览与遍历检查
        • 2.3.3.1、遍历检查
        • 2.3.3.2、整体总览

  
  
  

1、AVL树

1.1、基本概念介绍

  1)、AVL树是什么
  AVL树实际是对二叉搜索树的特化,又被称为高度平衡二叉搜索树。

  二叉搜索树问题说明: 数据有序或接近有序时,二叉搜索树将退化为单支树,查找元素相当于在顺序表中搜索元素,从而使得效率低下。
  解决方法: 当向二叉搜索树中插入新结点后,如果能保证每个结点的左右子树高度之差的绝对值不超过1(需要对树中的结点进行调整),即可降低树的高度,从而减少平均搜索长度。
  
  介绍: 一棵AVL树,其要么是空树,要么是具有以下性质的二叉搜索树:
  1、它的左右子树都是AVL树
  2、左右子树高度之差(简称平衡因子)的绝对值不超过1(-1、0、1)
  
  注意事项:
  1、当前根节点的平衡因子 = 右子树高度 - 左子树高度。平衡因子为非必需配置,可实现也可不实现。
在这里插入图片描述
  平衡因子==0,说明左右子树高度相等,平衡因子=1、-1,说明左右子树存在一边高一边低的现象。
  
  2、需要理解这里高度差的含义:这里必须保证任意一颗子树的高度差都不超过1。错误示范如下:
在这里插入图片描述

  3、由于结点个数原因,AVL树的高度平衡控制不能时刻保证左右子树高度随时相等,才有了这里平衡因子的存在。
  4、如果一棵二叉搜索树是高度平衡的,它就是AVL树。如果它有 N N N个结点,其高度可保持在 O ( l o g 2 N ) O(log_2 N) O(log2N),搜索时间复杂度O( l o g 2 N log_2 N log2N)。
  
  
  

1.2、重要框架与特性实现

1.2.1、如何搭建AVL树的结点 :AVLTreeNode,引入三叉链

  1)、三叉链
  介绍:结点中不仅存储左右孩子结点,还存储父结点。

  三叉链表是二叉树的另一种主要的链式存储结构。三叉链表与二叉链表的主要区别在于,它的结点比二叉链表的结点多一个指针域,该域用于存储一个指向本结点双亲的指针。

  说明:三叉链的这种存储方式,使得我们在实现AVL树的高度平衡时带来一定便捷,但同理的也会存在一些缺陷。(此部分可在后续实现时感受到)
  
  
  2)、AVLTreeNode实现
  AVL树的结点以三叉链实现,注意它的成员变量_bf,平衡因子不是非必须实现的。

template<class K,class V>
struct AVLTreeNode
{
	AVLTreeNode<K, V>* _left; //左孩子指针
	AVLTreeNode<K, V>* _right; //右孩子指针
	AVLTreeNode<K, V>* _parent; //父节点指针

	pair<K, V> _kv;//当前结点值
	int _bf;//平衡因子:balance factor

	AVLTreeNode(const pair<K,V>& kv)//构造函数
		:_left(nullptr)
		,_right(nullptr)
		,_parent(nullptr)
		,_kv(kv)
		,_bf(0)//新增一个结点,其平衡因子初始时为0
	{}
};

  
  
  
  

1.2.2、如何搭建AVL树

  1)、框架搭建
  基本框架如下:

template<class K,class V>
class AVLTree
{
	typedef AVLTreeNode<K, V> Node;
public:
	//其它函数实现
	
private:
	Node* _root = nullptr;
};

  
  
  

1.2.3、AVL树插入讲解

  与二叉搜索树相比,AVL树在其基础上引入了平衡因子。在插入数据后,AVL除了要满足二叉搜索树的基本特性(左子树比根结点小,右子树比根结点大),也要满足自己的特性(每个结点的左右子树高度之差的绝对值不超过1)。
  考虑到上述情况,AVL树的插入过程可以分为两步:
  1、按照二叉搜索树的方式插入新节点;
  2、调整节点的平衡因子;
  
  
  

1.3、AVL树插入

1.3.1、step1:按照二叉搜索树的方式插入新节点

  1)、基础说明
  AVL树插入新结点整体思路与二叉搜索树一致,但在细节上有一定区别。
  1、待插入值左右子树的位置判断:const pair<K, V>& kv,可以看到,我们给定的参数是一个键值对,其含有两个变量参数first、second。根据我们之前对set、map的学习,在pair判断左右子树时,我们同一看第一参数first,即key值。故而比较的是cur->_kv.first kv.first
  2、三叉链带来的链接关系变化:对单个结点,当前新增结点的左右孩子指针_left_right在构造函数中已经处理完成,但需要处理其父节点指针_parent与原先树的关系。同理,也要处理父节点中左右孩子与新增结点的关系。

	bool Insert(const pair<K, V>& kv)
	{
		//step1:按照二叉搜索树的方式插入新节点
		Ⅰ、查找合适的插入位置
		if (_root == nullptr)
		{
			_root = new Node(kv);
			return true;
		}

		Node* cur = _root;
		Node* 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//cur->kv.first==kv.first,key值已存在
			{
				return false;
			}
		}

		Ⅱ、当出了循环,说明cur==nullptr,找到合适位置,开始插入
		cur = new Node(kv);
		cur->_parent = parent;
		if (parent->_kv.first < kv.first)
			parent->_right = cur;
		else//parent->_kv.first > kv.first
			parent->_left = cur;


		//step2:调整节点的平衡因子
		//此部分后续讲述
	}

  
  
  
  

1.3.2、step2:调整节点的平衡因子

1.3.2.1、更新平衡因子的相关规则

  1)、为什么要更新平衡因子
  说明: 当新结点插入后,AVL树的平衡性可能会遭到破坏,此时就需要更新平衡因子,并检测是否破坏了AVL树的平衡性。
  PS: 只有新增结点的祖先结点,其平衡因子可能会受到影响(父节点的平衡因子一定受到影响,需要调整,再往上的祖先结点不一定受影响)。所以我们需要沿着当前新增结点,往上找其祖先结点(包含父结点),不断更新平衡因子。
在这里插入图片描述

  
  2)、如何更新平衡因子
  1、对单次结点进行更新:
  ①若新增结点在左,则parent->_bf--;
  ②若新增结点在右,则parent->_bf++;
在这里插入图片描述

  
  2、判断是否沿祖先迭代更新:
  ①更新后,若parent->_bf==0,则不必向上更新。 该情况说明在插入前parent的平衡因子为-1 or 1,即左右子树一边高一边低。新结点插入在了低的子树上,使得插入后parent左右子树高度不变,故而不用往上更新。
在这里插入图片描述

  ②更新后,若parent->_bf ==1 or -1,则需要向上更新。 该情况说明在插入前parent的平衡因子为0,即左右子树高度相等。新结点的插入使得parent左右子树高度改变。

  ③ 更新后,若parent->_bf== 2 or -2, 则平衡打破,需要旋转。 该情况说明在插入前parent的平衡因子是1 or -1, 已经处于平衡临界值,新节点的插入使得平衡因子变为 2 or -2,故而需要对当前parent所在的子树进行旋转处理,以维持高度平衡。
在这里插入图片描述

  ④更新后,若parent->_bf >2 or <-2,不存在。若我们得到结果如此,只能说明之前AVL树的平衡操作出了问题,需要进行检查。

  
  
  3)、相关代码实现
  代码如下:

		//step2:调整节点的平衡因子
		while (parent)
		{
			Ⅰ、对单次结点进行更新
			if (parent->_left == cur)
				parent->_bf--;//新增在左
			else
				parent->_bf++;//新增在右

			Ⅱ、判断是否需要沿祖先迭代更新
			if (0 == parent->_bf)
			{	//更新后,若parent->_bf==0,则不必向上更新。
				break;
			}
			else if (1 == abs(parent->_bf))
			{
				//更新后,若parent->_bf == 1 or -1,则需要向上更新。
				cur = parent;
				parent = parent->_parent;
			}
			else if (2 == abs(parent->_bf))
			{
				//更新后,若parent->_bf== 2 or -2, 则平衡打破,需要旋转。
				//此部分后续讲述
			}
			else
			{
				assert(false);//直接断言报错
			}

		}

  
  
  

1.3.2.2、AVL树旋转模式(四类)

  根据之前小节分析,AVL树中插入一个新节点,在有些情况下会使得当前树不平衡,此时就必须调整树的结构,使之平衡化。根据节点插入位置的不同,AVL树的旋转分为四种。

1.3.2.2.1、左单旋:新结点插入较高右子树的右侧(右右)

在这里插入图片描述

  相关代码如下:

				if (parent->_bf == 2 && cur->_bf == 1)//左单旋
				{
					RotateL(parent);
					break;
				}
private:
	void RotateL(Node* parent)
	{
		Node* subR = parent->_right;
		Node* subRtoL = subR->_left;
		Node* grendparent = parent->_parent;

		//修改链接关系以达成旋转
		parent->_right = subRtoL;
		parent->_parent = subR;

		if (subRtoL)//h>=0,subRtoL可能不存在
			subRtoL->_parent = parent;

		subR->_left = parent;
		if (_root == parent)//parent为原先AVL树的根节点
		{
			subR->_parent = nullptr;
			_root = subR;
		}
		else//parent为原先AVL树的分支节点
		{
			subR->_parent = grendparent;
			if (grendparent->_left == parent)
				grendparent->_left = subR;
			else
				grendparent->_right = subR;
		}

		//处理修改后的平衡因子
		subR->_bf = parent->_bf = 0;
	}

  示意图:
在这里插入图片描述

  
  

1.3.2.2.2、右单旋:新结点插入较高左子树的左侧(左左)

在这里插入图片描述

  相关代码如下:

				else if (parent->_bf == -2 && cur->_bf == -1)//右单旋
				{
					RotateR(parent);
				}
	void RotateR(Node* parent)
	{
		Node* subL = parent->_left;
		Node* subLtoR = subL->_right;
		Node* grendparent = parent->_parent;

		//处理链接关系
		parent->_left = subLtoR;
		parent->_parent = subL;

		if (subLtoR)
			subLtoR->_parent = parent;

		subL->_right = parent;
		if (_root = parent)
		{
			subL->_parent = nullptr;
			_root = subL;
		}
		else
		{
			subL->_parent = grendparent;
			if (grendparent->_left == parent)
				grendparent->_left = subL;
			else
				grendparent->_right = subL;
		}
		//处理平衡因子
		subL->_bf = parent->_bf = 0;
	}

  示意图:
在这里插入图片描述

  
  
  
  

1.3.2.2.3、左右双旋:新结点插入较高左子树的右侧(左右)

  即先左单旋再右单旋。如下图,在b、c位置插入,b or c 的高度改变,都是改变较高左子树的右侧,其都会引发双旋。

在这里插入图片描述

  
  相关代码如下:

				else if (parent->_bf == -2 && cur->_bf == 1)//左右双旋
				{
					RotateLR(parent);
				}
	void RotateLR(Node* parent)
	{
		Node* subL = parent->_left;
		Node* subLtoR = subL->_right;
		int bf = subLtoR->_bf;

		//左右双旋:先左旋,再右旋
		RotateL(subL);
		RotateR(parent);
		
		//判断情况,调整平衡因子
		subLtoR->_bf = 0;
		if (bf == 0)//插入subLtoR节点本身
		{
			parent->_bf = subL->_bf = 0;
		}
		else if (bf = -1)//插入在subLtoR左子树
		{
			parent->_bf = 1;
			subL->_bf = 0;
		}
		else if (bf = 1)//插入在subLtoR右子树
		{
			parent->_bf = 0;
			subL->_bf = -1;
		}
		else
		{
			assert(false);
		}
	}

  
  示意图:
在这里插入图片描述
  
  关于平衡因子的调节解释:
在这里插入图片描述
  
  
  
  

1.3.2.2.4、右左双旋:新结点插入较高右子树的左侧(右左)

  先右单旋再左单旋
在这里插入图片描述

  相关代码如下:

				else if(parent->_bf==2 && cur->_bf==-1)//右左双旋
				{
					RotateRL(parent);
				}
	void RotateRL(Node* parent)
	{
		Node* subR = parent->_right;
		Node* subRtoL = subR->_left;
		int bf = subR->_bf;

		//右左双旋:先右单旋再左单旋
		RotateR(subR);
		RotateL(parent);

		//判断情况,调节平衡因子
		subRtoL->_bf = 0;
		if (bf == 0)//插入subRtoL节点本身
		{
			subR->_bf = parent->_bf = 0;
		}
		else if (bf == -1)//插入在subRtoL的左子树
		{
			parent->_bf = 0;
			subR->_bf = 1;
		}
		else if (bf == -1)//插入在subrtoL的右子树
		{
			parent->_bf = -1;
			subR->_bf = 0;
		}
		else
		{
			assert(false);
		}
	}

  
  示意图:
在这里插入图片描述

  关于平衡因子的调节解释:
在这里插入图片描述

  
  

1.3.3、整体实现总览与遍历检查

1.3.3.1、遍历检查

  
  1)、中序遍历检查是否满足二叉搜索树
  说明:如果中序遍历可得到一个有序的序列,就说明为二叉搜索树。但是不能证明其一定就满足AVL树。

void testAV01()
{
	int a[] = { 4, 2, 6, 1, 3, 5, 15, 7, 16, 14 };  // 测试双旋平衡因子调节
	//int a[] = { 16, 3, 7, 11, 9, 26, 18, 14, 15 };
	AVLTree<int, int> t1;
	for (auto e : a)
	{
		t1.Insert(make_pair(e, e));
	}

	t1.InOrder();
}

在这里插入图片描述

  
  
  2)、检查是否满足AVL树
  说明:我们可以实现一个求当前结点树的高度的函数,用于获取每个结点的高度差,即平衡因子,若其绝对值不超过2,则说明树是平衡树。
  注意:
  1、此处不能直接使用平衡因子来判断,因为上述实现中,平衡因子是我们手动更新的,无符保证真实实现的树和我们理论上的值相同(即当我们实现错误的时候)。
  2、如果我们实现了平衡因子,那么可以顺带检查一下我们手动更新的平衡因子是否计算正确。
  
  相关代码如下:

public:
	bool IsBalance()
	{
		return _IsBalance(_root);
	}
	
private:

	bool _IsBalance(Node* root)
	{
		//若树为空,满足AVL树
		if (root == nullptr)
			return true;

		//判断当前节点的平衡因子是否满足(左右子树高度差<2)
		int leftHeight = Height(root->_left);//求左子树高度
		int rightHeight = Height(root->_right);
		int diff = rightHeight - leftHeight;

		if (diff != root->_bf)
		{//这是为了判断实际树的平衡因子和我们手动更新的平衡因子是否匹配
			cout << root->_kv.first << ":当前结点平衡因子异常" << endl;
			return false;
		}

		//AVL树要满足每个结点的平衡因子都满足高度之差的绝对值不超过1
		return (abs(diff) < 2) 
			&& (_IsBalance(root->_left)) 
			&& (_IsBalance(root->_right));
	}

	int Height(Node* root)//求树的高度
	{
		if (root == nullptr)
			return 0;
		return max(Height(root->_left), Height(root->_right)) + 1;
	}

  
  

//可以使用随机值测试:丰富测试用例
void testAV02()
{
	size_t N = 10000;
	srand(time(0));
	AVLTree<int, int> t1;
	for (size_t i = 0; i < N; ++i)
	{
		int x = rand();
		t1.Insert(make_pair(x, i));
	}
	cout << "IsBalance:" << t1.IsBalance() << endl;
}

  
  

1.3.3.2、整体展示
#pragma once
#include<assert.h>
#include<iostream>
using namespace std;

template<class K,class V>
struct AVLTreeNode
{


	AVLTreeNode<K, V>* _left; //左孩子指针
	AVLTreeNode<K, V>* _right; //右孩子指针
	AVLTreeNode<K, V>* _parent; //父节点指针

	pair<K, V> _kv;//当前结点值,注意这里是键值对
	int _bf;//平衡因子:balance factor

	AVLTreeNode(const pair<K,V>& kv)//构造函数
		:_left(nullptr)
		,_right(nullptr)
		,_parent(nullptr)
		,_kv(kv)
		,_bf(0)//新增一个结点,其平衡因子初始时为0
	{}
};



template<class K,class V>
class AVLTree
{
	typedef AVLTreeNode<K, V> Node;
public:


	
	bool Insert(const pair<K, V>& kv)
	{
		//step1:按照二叉搜索树的方式插入新节点
		Ⅰ、查找合适的插入位置
		if (_root == nullptr)
		{
			_root = new Node(kv);
			return true;
		}

		Node* cur = _root;
		Node* 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//cur->kv.first==kv.first,key值已存在
			{
				return false;
			}
		}

		Ⅱ、当出了循环,说明cur==nullptr,找到合适位置,开始插入
		cur = new Node(kv);
		cur->_parent = parent;
		if (parent->_kv.first < kv.first)
			parent->_right = cur;
		else//parent->_kv.first > kv.first
			parent->_left = cur;


		//step2:调整节点的平衡因子
		while (parent)
		{
			Ⅰ、对单次结点进行更新
			if (parent->_left == cur)
				parent->_bf--;//新增在左
			else
				parent->_bf++;//新增在右

			Ⅱ、判断是否需要沿祖先迭代更新
			if (0 == parent->_bf)
			{	//更新后,若parent->_bf==0,则不必向上更新。
				break;
			}
			else if (1 == abs(parent->_bf))
			{
				//更新后,若parent->_bf == 1 or -1,则需要向上更新。
				cur = parent;
				parent = parent->_parent;
			}
			else if (2 == abs(parent->_bf))
			{
				//更新后,若parent->_bf== 2 or -2, 则平衡打破,需要旋转。
				
				if (parent->_bf == 2 && cur->_bf == 1)//左单旋
				{
					RotateL(parent);
				}
				else if (parent->_bf == -2 && cur->_bf == -1)//右单旋
				{
					RotateR(parent);
				}
				else if (parent->_bf == -2 && cur->_bf == 1)//左右双旋
				{
					RotateLR(parent);
				}
				else if (parent->_bf == 2 && cur->_bf == -1)//右左双旋
				{
					RotateRL(parent);
				}
				else
				{
					assert(false);
				}

				break;

			}
			else
			{
				assert(false);
			}

		}
		return true;

	}

	void InOrder()
	{
		_InOrder(_root);
		cout << endl;
	}

	bool IsBalance()
	{
		return _IsBalance(_root);
	}
private:

	bool _IsBalance(Node* root)
	{
		//若树为空,满足AVL树
		if (root == nullptr)
			return true;

		//判断当前节点的平衡因子是否满足(左右子树高度差<2)
		int leftHeight = Height(root->_left);//求左子树高度
		int rightHeight = Height(root->_right);
		int diff = rightHeight - leftHeight;

		if (diff != root->_bf)
		{//这是为了判断实际树的平衡因子和我们手动更新的平衡因子是否匹配
			cout << root->_kv.first << ":当前结点平衡因子异常" << endl;
			return false;
		}

		//AVL树要满足每个结点的平衡因子都满足高度之差的绝对值不超过1
		return (abs(diff) < 2) 
			&& (_IsBalance(root->_left)) 
			&& (_IsBalance(root->_right));
	}

	int Height(Node* root)//求树的高度
	{
		if (root == nullptr)
			return 0;
		return max(Height(root->_left), Height(root->_right)) + 1;
	}

	void _InOrder(Node* root)
	{
		if (root == nullptr)
			return;
		_InOrder(root->_left);
		cout << root->_kv.first << "," << root->_kv.second << endl;
		_InOrder(root->_right);
	}

	void RotateL(Node* parent)
	{
		Node* subR = parent->_right;
		Node* subRtoL = subR->_left;
		Node* grendparent = parent->_parent;

		//修改链接关系以达成旋转
		parent->_right = subRtoL;
		parent->_parent = subR;

		if (subRtoL)//h>=0,subRtoL可能不存在
			subRtoL->_parent = parent;

		subR->_left = parent;
		if (_root == parent)//parent为原先AVL树的根节点
		{
			subR->_parent = nullptr;
			_root = subR;
		}
		else//parent为原先AVL树的分支节点
		{
			subR->_parent = grendparent;
			if (grendparent->_left == parent)
				grendparent->_left = subR;
			else
				grendparent->_right = subR;
		}
		//处理修改后的平衡因子
		subR->_bf = parent->_bf = 0;
	}


	void RotateR(Node* parent)
	{
		Node* subL = parent->_left;
		Node* subLtoR = subL->_right;
		Node* grendparent = parent->_parent;

		//处理链接关系
		parent->_left = subLtoR;
		parent->_parent = subL;

		if (subLtoR)
			subLtoR->_parent = parent;

		subL->_right = parent;
		if (_root == parent)
		{
			subL->_parent = nullptr;
			_root = subL;
		}
		else
		{
			subL->_parent = grendparent;
			if (grendparent->_left == parent)
				grendparent->_left = subL;
			else
				grendparent->_right = subL;
		}
		//处理平衡因子
		subL->_bf = parent->_bf = 0;
	}

	void RotateLR(Node* parent)
	{
		Node* subL = parent->_left;
		Node* subLtoR = subL->_right;
		int bf = subLtoR->_bf;

		//左右双旋:先左旋,再右旋
		RotateL(subL);
		RotateR(parent);
		
		//判断情况,调整平衡因子
		subLtoR->_bf = 0;
		if (bf == 0)//插入subLtoR节点本身
		{
			parent->_bf = 0;
			subL->_bf = 0;
		}
		else if (bf == -1)//插入在subLtoR左子树
		{
			parent->_bf = 1;
			subL->_bf = 0;
		}
		else if (bf == 1)//插入在subLtoR右子树
		{
			parent->_bf = 0;
			subL->_bf = -1;
		}
		else
		{
			assert(false);
		}
	}
	

	void RotateRL(Node* parent)
	{
		Node* subR = parent->_right;
		Node* subRtoL = subR->_left;
		int bf = subRtoL->_bf;

		//右左双旋:先右单旋再左单旋
		RotateR(subR);
		RotateL(parent);

		//判断情况,调节平衡因子
		subRtoL->_bf = 0;
		if (bf == 0)//插入subRtoL节点本身
		{
			parent->_bf = 0;
			subR->_bf = 0;
		}
		else if (bf == -1)//插入在subRtoL的左子树
		{
			parent->_bf = 0;
			subR->_bf = 1;
		}
		else if (bf == 1)//插入在subrtoL的右子树
		{
			parent->_bf = -1;
			subR->_bf = 0;
		}
		else
		{
			assert(false);
		}
	}

	Node* _root = nullptr;
};



  
  
  
  
  
  

2、红黑树

2.1、基本概念与性质

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

  
  
  2)、红黑树的性质说明

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

  
  说明: 在确保上述性质后,红黑树就能保证:其最长路径中结点个数不会超过最短路径结点个数的两倍
  原因说明:举例极端情况,假设最短路径结点总数为N,其全为黑色结点。那么其最长路径中,黑色结点数量最多有N个,此时,不违背性质3、4的情况下所获得的最长路径就是再插入N-1个红色结点,则最长路径总结点数为2N-1<2N。
在这里插入图片描述

  
  

2.2、重要框架与特性实现

2.2.1、如何搭建红黑树的结点

  1)、框架搭建
  基本说明:仍旧保持三叉链不变,只是此处我们需要引入结点颜色,这里以枚举来实现。

enum Colour
{
	RED,
	BLACK
};

template<class K,class V>
struct RBTreeNode
{
	RBTreeNode<K, V>* _left;
	RBTreeNode<K, V>* _right;
	RBTreeNode<K, V>* _parent;

	pair<K, V> _kv;//当前结点存储值
	Colour _col;

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

};

  问题:为什么 _col(RED) 默认结点为红色?
  回答:事实上新增一个结点,其为红色还是黑色都是由我们任意控制的。只是在红黑树中,新增一个结点,仍旧要满足红黑树的特性,如果结点是黑色,那么根据性质4,每条路径都将受到影响,后续调整起来相对不便;若新增结点为红色,整体影响范围相对较小。
在这里插入图片描述

  
  
  
  

2.2.2、如何搭建红黑树

  1)、框架搭建
  基本框架如下:

template<class K,class V>
class RBTree
{
	typedef RBTreeNode<K, V> Node;

public:
	//……

private:
	Node* _root = nullptr;
};

  
  
  
  

2.2.3、红黑树的插入讲解

  同AVL树,红黑树也是在二叉搜索树的基础上,加上一定特性,满足其平衡限制,因此红黑树的插入也可分为两步:
  1、按照二叉搜索树的方式插入新节点
  2、检测新节点插入后,红黑树的性质是否造到破坏
  
  

2.3、红黑树的插入

2.3.1、step1:按照二叉搜索树的方式插入新节点

  此处与AVL树实现基本一致,相关注意细节在该部分已说明。

	bool Insert(const pair<K, V>& kv)
	{
		//step1:按照二叉搜索树的方式插入新节点
		Ⅰ、寻找位置
		if (_root == nullptr)//单独处理:根节点为空时
		{
			_root = new Node(kv);
			_root->_col = BLACK;//根节点默认为黑色
			return true;
		}

		//根节点不为空:
		Node* cur = _root;
		Node* 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//cur->kv.first==kv.first,key值已存在
			{
				return false;
			}
		}

		Ⅱ、插入值,调整链接关系
		cur = new Node(kv);
		cur->_parent = parent;
		if (parent->_kv.first < kv.first)//cur插入在parent的右孩子处
			parent->_right = cur;
		else
			parent->_left = cur;//cur插入在parent的左孩子处
		cur->_col = RED;//将插入的结点设置为红色

		//step2:检查新增结点后,是否还满足红黑树性质
		//后续说明

	}

  

2.3.2、step2:检查新增结点后红黑树性质是否遭到破坏

  1)、三类模式分析


  情况一: cur为红,parent为红,grandparent为黑,uncle存在且为红。
  
在这里插入图片描述

  一些说明:
  1、parent为红,根据性质,grandparent一定为黑;
  2、cur可以是parent的左右任意孩子;
  3、cur为红存在两种情况,一是它本身即为新增的结点,二是它的子树中存在新增节点,不断调整迭代,到它这里变红。
  
  处理方法:
  1、将parent、uncle改为黑,grandparent改为红。
  2、判断当前grandparent,若为根,则变回黑色;否不是根,则迭代,把grandparent当成cur,继续向上调整。
  
  
  


  情况二: cur为红,p为红,g为黑,u不存在/u存在且为黑
  
在这里插入图片描述

  一些说明:
  1、若uncle不存在,则cur一定是新增结点,且abcde为空树。原因:反证,若cur不是新增结点,上述中cur、parent连续为红色,则二者必有一黑,以确保性质,但在uncle不存在的情况,这样又不满足性质。
  2、若uncle存在且为黑,则新增结点在cur的任意子树中。原因:uncle为黑,确保了每条路径黑结点必须有两个,新增结点我们设置为红色,若cur为新增,则不符合上述性质。即cur原先为黑结点,是由于其子树颜色迭代,导致它当前为红。
  
  处理方法: 旋转+变色。
  1、旋转grandparent:若parentgrandparent的左孩子,且curparent的左孩子(一条直线上),则对grandparent进行右单旋;若parentgrandparent的右孩子,且curparent的右孩子,则对grandparent进行左单旋
  2、变色:parent变黑,grandparent变红。
  
在这里插入图片描述

  PS:
  ①为什么情况一不适用:cur为红,若将parent变黑,无论uncle存在或者原先为黑,变相等于当前路径比其它路径多增加一个黑结点,违背了性质。(在情况一中,parent变黑,同时uncle也变黑,不会引发上述问题)
  ②上述这种情况处理完成后,实际上就不用继续向上迭代调整。因为此情况下调整结束后,就满足了红黑树的规则。
  
  
  


  情况三: cur为红,p为红,g为黑,u不存在/u存在且为黑
在这里插入图片描述

  
  一些说明:
  1、实则为情况二的变形,处理方法同样为旋转+变色,但与之区别在于此处需要双旋。
  2、uncle不存在,cur为新增,uncle存在且为黑,cur为黑,新增结点在cur任意孩子位置,向上迭代使得cur变红。
  
  处理方法: 旋转+变色。
  1、旋转grandparent:若parentgrandparent的左孩子,且curparent的右孩子,则对grandparent进行左右单旋;若parentgrandparent的右孩子,且curparent的右孩子,则对grandparent进行右左单旋
  2、变色:cur变黑,grandparent变红。
  
  

  2)、相关代码实现
  依照上述,总结可得,红黑树的关键是看uncle:

若uncle存在且为红,变色继续往上处理;
若uncle不存在/存在且为黑,则旋转(四类)+变色。

  基本框架如下:

		//step2:检查新增结点后,是否还满足红黑树性质
		while (parent && parent->_col == RED)//新结点后,无论是cur本身,还是迭代后变色,如果parent、cur颜色皆红,则违反性质3:不能有连续的红结点,因此需要调整
		{
			//固定:
			Node* grandparent = parent->_parent;
			assert(grandparent && grandparent->_col == BLACK);//grandparent存在且为黑

			//关键看叔叔,此处对uncle和parent在不同位置分别处理
			if (parent == grandparent->_left)
			{
				Node* uncle = grandparent->_right;
				//……
				//三种情况分别处理
			}
			else//parent == grandparent->_right
			{
				Node* uncle = grandparent->_left;
				//……
				//三种情况分别处理
			}

		}

  
  
  


2.3.3、整体实现总览与遍历检查

2.3.3.1、遍历检查

  1)、如何检查当前树是否为红黑树?
  方案一:看高度。
  分析:如果使用最长路径和最短路径高度差不超过二倍来比较。存在一个问题,满足该高度差,不一定代表其结点颜色满足红黑树需求。
  
  方案二:看结点颜色。
  1、检查根节点颜色为黑。
  2、检查红色结点不连续:遍历结点,若当前结点为红色,则比较其与其孩子结点。
  3、检查每条路径的黑结点数量。
  
  2)、写法说明

在这里插入图片描述写法一:


	bool IsBalance()
	{
		//1、检查根节点
		if (_root == nullptr)
			return true;
		if (_root->_col == RED)
		{
			cout << "根节点不满足黑色" << endl;
			return false;
		}
		
		//2、检查每条路径的黑色结点数目
		int blackNum = 0;//用于统计树中黑色结点数
		return PrevCheck(_root, blackNum);
	}

private:
	bool PrevCheck(Node* root, int blackNum)
	{
		if (root == nullptr)
		{
			cout << blackNum << endl;
			return true;
		}

		if (root->_col == BLACK)
			++blackNum;
		
		//3、检查红色结点是否连续	
		if (root->_col == RED && root->_parent->_col == RED)
		{
			cout << "存在连续的红色结点" << endl;
			return false;
		}

		return PrevCheck(root->_left, blackNum)
			&& PrevCheck(root->_right, blackNum);
	}

  演示结果:
在这里插入图片描述

  问题说明:
  此处我们遍历到每条路径末尾时,将当前路径的黑色结点树统计并打印出来,这样存在一个问题,需要我们核对每条路径的黑色结点是否匹配,这样当路径很多时,我们手动核对处理起来麻烦,不具有自动和高效性。

		if (root == nullptr)
		{
			cout << blackNum << endl;
			return true;
		}

  因此需要对其修改。
  
  

在这里插入图片描述写法二:

	bool IsBalance()
	{
		if (_root == nullptr)
			return true;
		if (_root->_col == RED)
		{
			cout << "根节点不满足黑色" << endl;
			return false;
		}

		//以红黑树中任意一条路径的黑色结点数作为基准值,其余路径的黑色结点与其比较
		int basicalNum = 0;
		Node* cur = _root;
		while (cur)
		{
			if (cur->_col == BLACK)
				++basicalNum;
			cur = cur->_left;//此处这种写法是以红黑树中最左边的路径作为基准路径
		}
		int blackNum = 0;//用于统计树中黑色结点数
		return PrevCheck(_root, blackNum, basicalNum);
	}

private:
	bool PrevCheck(Node* root, int blackNum, int basicalNum)
	{
		if (root == nullptr)
		{
			if (blackNum == basicalNum)
				return true;
			else
				{
					cout << "某条路径黑色结点数不匹配" << endl;
					return false;
				}
		}

		if (root->_col == BLACK)
			++blackNum;
		if (root->_col == RED && root->_parent->_col == RED)
		{
			cout << "存在连续的红色结点" << endl;
			return false;
		}

		return PrevCheck(root->_left, blackNum, basicalNum)
			&& PrevCheck(root->_right, blackNum, basicalNum);
	}

  
  

在这里插入图片描述写法三:

	bool IsBalance()
	{
		if (_root == nullptr)
			return true;
		if (_root->_col == RED)
		{
			cout << "根节点不满足黑色" << endl;
			return false;
		}

		int BasicalNum = 0;//用于确定基准值:前序遍历时的第一条路径
		int blackNum = 0;//用于统计树中黑色结点数
		return PrevCheck(_root, blackNum, BasicalNum);
	}

private:
	bool PrevCheck(Node* root, int blackNum, int& basicalNum)
	{
		if (root == nullptr)
		{
			if (basicalNum == 0)
			{
				basicalNum == blackNum;
				return true;
			}
			else
			{
				if (blackNum == basicalNum)
					return true;
				else
				{
					cout << "某条路径黑色结点数不匹配" << endl;
					return false;
				}
			}
		}

		if (root->_col == BLACK)
			++blackNum;
		if (root->_col == RED && root->_parent->_col == RED)
		{
			cout << "存在连续的红色结点" << endl;
			return false;
		}

		return PrevCheck(root->_left, blackNum, basicalNum)
			&& PrevCheck(root->_right, blackNum, basicalNum);
	}

  
  
  

2.3.3.2、整体总览

  
  

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



enum Colour
{
	RED,
	BLACK
};

template<class K,class V>
struct RBTreeNode
{
	RBTreeNode<K, V>* _left;
	RBTreeNode<K, V>* _right;
	RBTreeNode<K, V>* _parent;

	pair<K, V> _kv;//当前结点存储值
	Colour _col;

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

};

template<class K,class V>
class RBTree
{
	typedef RBTreeNode<K, V> Node;

public:
	bool Insert(const pair<K, V>& kv)
	{
		//step1:按照二叉搜索树的方式插入新节点
		Ⅰ、寻找位置
		if (_root == nullptr)//单独处理:根节点为空时
		{
			_root = new Node(kv);
			_root->_col = BLACK;//根节点默认为黑色
			return true;
		}

		//根节点不为空:
		Node* cur = _root;
		Node* 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//cur->kv.first==kv.first,key值已存在
			{
				return false;
			}
		}

		Ⅱ、插入值,调整链接关系
		cur = new Node(kv);
		cur->_parent = parent;
		if (parent->_kv.first < kv.first)//cur插入在parent的右孩子处
			parent->_right = cur;
		else
			parent->_left = cur;//cur插入在parent的左孩子处
		cur->_col = RED;//将插入的结点设置为红色

		//step2:检查新增结点后,是否还满足红黑树性质
		while (parent && parent->_col == RED)//新结点后,无论是cur本身,还是迭代后变色,如果parent、cur颜色皆红,则违反性质3:不能有连续的红结点,因此需要调整
		{
			//固定:
			Node* grandparent = parent->_parent;
			assert(grandparent && grandparent->_col == BLACK);//grandparent存在且为黑

			//关键看叔叔,此处对uncle和parent在不同位置分别处理
			if (parent == grandparent->_left)
			{
				Node* uncle = grandparent->_right;
				
				//情况一
				if (uncle&& uncle->_col == RED)//叔叔存在且为红
				{
					//处理方法:变色
					parent->_col = uncle->_col = BLACK;
					grandparent->_col = RED;
					//继续向上调整
					cur = grandparent;
					parent = cur->_parent;
				}
				else //叔叔不存在或叔叔存在且为黑:情况二+情况三
				{
					//处理方法:旋转+变色(旋转有四类)

					//Ⅰ:情况二:p在g左,c在p左,成直线。右单旋+g变色为红,p变色为黑
					if (cur == parent->_left)
					{
						RotateR(grandparent);
						grandparent->_col = RED;
						parent->_col = BLACK;
					}
					else//Ⅱ:情况三:p在g左,c在p右,成折线。左右双旋+g变色为红,c变色为黑
					{
						RotateL(parent);
						RotateR(grandparent);
						grandparent->_col = RED;
						cur->_col = BLACK;
					}
					break;//情况二、情况三完成调整后,满足红黑树规则,不必继续向上调整
				}
			}
			else//parent == grandparent->_right
			{
				Node* uncle = grandparent->_left;

				//情况一
				if (uncle&& uncle->_col == RED)//叔叔存在且为红
				{
					//处理方法:变色
					parent->_col = uncle->_col = BLACK;
					grandparent->_col = RED;
					//继续向上调整
					cur = grandparent;
					parent = cur->_parent;
				}
				else //叔叔不存在或叔叔存在且为黑:情况二+情况三
				{
					//处理方法:旋转+变色(旋转有四类)

					//Ⅰ:情况二:p在g右,c在p右,成直线。左单旋+g变色为红,p变色为黑
					if (cur == parent->_right)
					{
						RotateL(grandparent);
						grandparent->_col = RED;
						parent->_col = BLACK;
					}
					else//Ⅱ:情况三:p在g右,c在p左,成折线。右左双旋+g变色为红,c变色为黑
					{
						RotateR(parent);
						RotateL(grandparent);
						grandparent->_col = RED;
						cur->_col = BLACK;
					}
					break;//情况二、情况三完成调整后,满足红黑树规则,不必继续向上调整
				}
			}

		}
		_root->_col = BLACK;

	}

	void InOrder()
	{
		_InOrder(_root);
		cout << endl;
	}

	bool IsBalance()
	{
		if (_root == nullptr)
			return true;
		if (_root->_col == RED)
		{
			cout << "根节点不满足黑色" << endl;
			return false;
		}

		写法一:
		以红黑树中任意一条路径的黑色结点数作为基准值,其余路径的黑色结点与其比较
		//int basicalNum = 0;
		//Node* cur = _root;
		//while (cur)
		//{
		//	if (cur->_col == BLACK)
		//		++basicalNum;
		//	cur = cur->_left;//此处这种写法是以红黑树中最左边的路径作为基准路径
		//}

		int BasicalNum = 0;//用于确定基准值:前序遍历时的第一条路径
		int blackNum = 0;//用于统计树中黑色结点数
		return PrevCheck(_root, blackNum, BasicalNum);
	}

private:
	bool PrevCheck(Node* root, int blackNum, int& basicalNum)
	{
		if (root == nullptr)
		{
			if (basicalNum == 0)
			{
				basicalNum == blackNum;
				return true;
			}
			else
			{
				if (blackNum == basicalNum)
					return true;
				else
				{
					cout << "某条路径黑色结点数不匹配" << endl;
					return false;
				}
			}
		}

		if (root->_col == BLACK)
			++blackNum;
		if (root->_col == RED && root->_parent->_col == RED)
		{
			cout << "存在连续的红色结点" << endl;
			return false;
		}

		return PrevCheck(root->_left, blackNum, basicalNum)
			&& PrevCheck(root->_right, blackNum, basicalNum);
	}

	void _InOrder(Node* root)
	{
		if (root == nullptr)
			return;
		_InOrder(root->_left);
		cout << root->_kv.first << "," << root->_kv.second << endl;
		_InOrder(root->_right);
	}

	void RotateL(Node* parent)
	{
		Node* subR = parent->_right;
		Node* subRtoL = subR->_left;
		Node* grendparent = parent->_parent;

		//修改链接关系以达成旋转
		parent->_right = subRtoL;
		parent->_parent = subR;

		if (subRtoL)//h>=0,subRtoL可能不存在
			subRtoL->_parent = parent;

		subR->_left = parent;
		if (_root == parent)//parent为原先AVL树的根节点
		{
			subR->_parent = nullptr;
			_root = subR;
		}
		else//parent为原先AVL树的分支节点
		{
			subR->_parent = grendparent;
			if (grendparent->_left == parent)
				grendparent->_left = subR;
			else
				grendparent->_right = subR;
		}
	}


	void RotateR(Node* parent)
	{
		Node* subL = parent->_left;
		Node* subLtoR = subL->_right;
		Node* grendparent = parent->_parent;

		//处理链接关系
		parent->_left = subLtoR;
		parent->_parent = subL;

		if (subLtoR)
			subLtoR->_parent = parent;

		subL->_right = parent;
		if (_root == parent)
		{
			subL->_parent = nullptr;
			_root = subL;
		}
		else
		{
			subL->_parent = grendparent;
			if (grendparent->_left == parent)
				grendparent->_left = subL;
			else
				grendparent->_right = subL;
		}
	}

	Node* _root = nullptr;
};

  
  
  
  
  
  
  
  

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

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

相关文章

超越ChatGPT?新发布:“讯飞星火认知大模”到底行不行?

国内又一巨头发布 大语言模型&#xff0c;是 PPT 融资还是真材实料 &#xff1f; 作为程序员&#xff0c;到底面对这一趋势&#xff0c;我们何去何从 &#xff1f; 目录 讯飞星火&#xff0c;5月6日如约而至 一、你真的了解科大讯飞吗&#xff1f; 二、讯飞星火大模型将超越…

C++ 面向对象特征4 多态(Polymorphism)

目录 1、对多态的理解 2、实现多态的例子 3、多态的意义 4、静态联编与动态联编 1、对多态的理解 同一对象可以有多重层级递进身份 在不同的场合中&#xff0c;被外界所关注的是不同的身份&#xff0c;但本质和应有的行为并不会因外界眼光而改变。 比如说我自己 kali-Myon…

Dev C++中出现 undefined reference to XXX 错误的解决方式

出现 undefined reference to XXX 错误的现象&#xff1a; 主函数中调用在其他文件中定义的函数&#xff0c;编译报错&#xff1a;未定义的引用xxx。 原理&#xff1a;编译器在生成可执行文件的过程包括预处理、编译、汇编、链接&#xff0c;这4个过程&#xff0c;这个问题一般…

【AI】YOLOv3原理详解

1、前言 YOLOv1~3作者是约瑟夫雷德蒙(Joseph Chet Redmon),他的网站:https://pjreddie.com/ YOLOv1网站:https://pjreddie.com/darknet/yolov1/ YOLOv2网站:https://pjreddie.com/darknet/yolov2/ YOLOv3网站:https://pjreddie.com/darknet/yolo/ YOLOv4作者是Alexeyab…

KingbaseES V8R6 集群运维案例--备库timeline not contain minimum recovery point故障

​ 案例现象&#xff1a; KingbaseES V8R6集群备库启动后&#xff0c;加入集群失败&#xff0c;sys_log日志信息提示&#xff0c;如下图所示&#xff1a; 适用版本&#xff1a;kingbaseES V8R6 一、问题分析 在timeline对应的history文件中会记录每次timeline切换时所对应的lsn…

【C++起飞之路】初级——命名空间、输入输出流

C&#xff1a;命名空间、输入输出流 一、命名空间1、命名空间的定义2、命名空间中成员的使用非嵌套命名空间&#xff1a;嵌套命名空间&#xff1a; 3、命名空间 三种展开方式<1>指定命名空间访问<2> using声明a、using引入某个成员b、using展开命名空间❗注意&…

离散化(算法)

目录 一、离散化的概念二、离散化的模板三、离散化的应用题目思路分析代码实现 一、离散化的概念 离散化是一种将连续数据映射到离散值的过程。它通常用于优化某些算法&#xff0c;尤其是与区间查询相关的问题。 在离散化过程中&#xff0c;我们将一组实数转换为一组整数&#…

美团企业版背后,费控和「know-how」的双重苦战

在企业消费管理的赛道里&#xff0c;美团不能算新手&#xff0c;但客观来看&#xff0c;也并不算是熟练的玩家。对这家中国骨灰级的玩家而言&#xff0c;这不仅是一场费控报销的仗&#xff0c;更是一个从C到B的观念转型之路。 作者|思杭 编辑|皮爷 出品|产业家 美团再出招。 …

从零构建自己的脚手架

从零构建自己的脚手架 简介 什么是CLI CLI 全称是 Command Line Interface&#xff0c;是一类通过命令行交互的终端工具。日常工作中常用的脚手架有 vue-cli、create-react-app、angular-cli 等&#xff0c;都是通过简单的初始化命令&#xff0c;完成内容的快速构建。 为什…

Unity Audio -- (3)创建3D音效

本节会添加场景中小瀑布的音效。小瀑布的音效会有一个作用范围&#xff0c;也会根据角色所处的位置不同&#xff0c;产生不同的效果。 添加小瀑布的声音 1. 在Hierarchy中&#xff0c;点击右键&#xff0c;选择Audio -> Create Audio Source&#xff0c;将这个新的Audio So…

HEVC学习之去方块滤波

一、概要 视频编码为视频带来的压缩伪影呈现出的效果各有不同&#xff0c;但其原因总结起来为高频信息的失真以及基于块的编码消除了块与块之间的相似性。 为了弥补基于块的编码带来的影响&#xff0c;HEVC中引入了去方块滤波。 AVC中采取对44块的边界进行去方块滤波&#xf…

通过自定义域名 + SSL 的方式访问 Amazon MQ for RabbitMQ

引言&#xff1a; 一般为了解决应用解耦&#xff0c;异步处理&#xff0c;流量削峰等问题&#xff0c;实现高性能&#xff0c;高可用&#xff0c;可伸缩和最终一致性的架构&#xff1b;我们会引入消息队列中间件来完善架构设计。 对于需要消息传递协议的应用程序&#xff0c;…

CCFCSP 201409-2 画图

思路上很容易想到即使用一个标记数组对上过色的模块进行标记&#xff0c;最后遍历该数组得到被标记的模块数即可 #include<iostream>using namespace std;int mapp[105][105]; int ans0;int main(){int n;cin>>n;for(int i0;i<n;i){int x1,y1,x2,y2;cin>>…

【P15】JMeter 正则表达式提取器(Regular Expression Extractor)

文章目录 一、准备工作二、测试计划设计 一、准备工作 慕慕生鲜&#xff1a; http://111.231.103.117/#/login 进入网页后&#xff0c;登录&#xff0c;页面提供了账户和密码 搜索框输入“虾” 右键检查或按F12&#xff0c;打开调试工具&#xff0c;点击搜索 二、测试计划设…

【Java项目】SpringCloud项目注册到Nacos中心时显示的是内网IP导致不同服务器之间无法互相调用接口的解决并发

微服务项目地址—动动你发财的小手点一个stars吧 出现这个问题是之前我也就遇到过的&#xff0c;这个问题的情况就是&#xff1a; 我们知道微服务项目是可以把不同的项目部署在不同的服务器上从而减少某一台服务器的压力&#xff0c;我们只需要为每一个服务配置一个注册中心即…

计算机基础--计算机存储单位

一、介绍 计算机中表示文件大小、数据载体的存储容量或进程的数据消耗的信息单位。在计算机内部&#xff0c;信息都是釆用二进制的形式进行存储、运算、处理和传输的。信息存储单位有位、字节和字等几种。各种存储设备存储容量单位有KB、MB、GB和TB等几种。 二、基本存储单元…

【进阶知识】显示管理器,窗口管理器,桌面环境/桌面管理器,显示服务器

文章目录 一、显示管理器&#xff08;Display Manager&#xff09;1.1 什么是 Linux 中的显示管理器&#xff1f;1.2 不同的显示管理器1.3 其他控制台显示管理器图形界面显示管理器 二、窗口管理器&#xff08;Window Manager&#xff09;三、桌面环境/桌面管理器&#xff08;D…

AD9680之JESD204B接口2路、4路、8路的14bit 500MSPS/1GSPS/1.25GSPS采样率子卡的中文版本设计及调试经验资料分享

板卡概述&#xff1a; 【FMC155】 FMC155 是一款基于 VITA57.1 标准的&#xff0c;实现 2 路 14-bit、500MSPS/1GSPS/1.25GSPS 直流耦合 ADC 同步采集 FMC 子卡模 块。 该模块遵循 VITA57.1 规范&#xff0c;可直接与 FPGA 载卡配合使用&#xff0c;板 卡 ADC 器件采用 ADI 的…

MySQL基础篇补充 | 单行函数(数值函数、字符串函数、日期函数、流程控制函数、加密与解密函数、MySQL信息函数)

目录 一&#xff1a;单行函数 1. 数值函数 &#xff08;1&#xff09;基本函数 &#xff08;2&#xff09;角度与弧度互换函数 &#xff08;3&#xff09;三角函数 &#xff08;4&#xff09;指数与对数 &#xff08;5&#xff09;进制间的转换 2. 字符串函数 3. 日期和…

Nginx配置浏览器缓存,页面展示更快一步

1.简介 缓存能够存储请求的响应结果&#xff0c;可以很方便的再次访问&#xff0c;使用缓存的优点是很明显的。 加速内容的访问&#xff0c;降低响应时间减少服务器的负载 Nginx不仅仅是一个web服务器&#xff0c;它也是一个web缓存服务器。通过Nginx缓存&#xff0c;我们对…