【数据结构】图论与并查集

news2025/1/19 20:37:49

一、并查集

1.原理

  1. 简单的讲并查集,就是查询两个个元素,是否在一个集合当中,这里的集合的形式进行表示。
  2. 并查集的本质就是森林, 即多棵树。

 我们再来简单的举个例子:

  • 假设此时的你是大一新生,刚进入大学,肯定是先找到宿舍在哪里,然后跟寝室里面的舍友互相认识一下,先形成一个小团体。
  • 假设,宿舍总共6个人,也就是6个人的集合。几乎所有的大学生都是这样先跟周围的人进行联系起来的。
  • 然后辅导员召集班会,这时的你欣然前往,并在讲台上自信的介绍自己,然后吸引或者主动又认识了一群人。这时你或许又跟其它的人进行了关联,或成为了好友,或成为了恋人……

下面我们用如上例子进行展开讨论:

  • 宿舍六人,即六个人,如何判断两个人在同一个集合? 如何进行实现?
  1. 先来解决第一个问题,六个人,选出一个宿舍长,只要两个人的宿舍长是一样的,即可判断两个人在一个集合。
  2. 再来解决第二个问题,既然宿舍长有了,我们都与这个宿舍长产生关联即可,即用树的形式进行表示,至于如何表示,我们可以用双亲表示法进行表示,即每个人记住其宿舍长的名字即可。更为形象的我们可以用下图进行表示:
  3. 更进一步,如何用计算机存储这种结构呢?我们只需对每个人名生成一个下标连续,用计算机进行存储即可。用下图进行直观的理解:
    在这里插入图片描述
  4. 对这张图我们再说明一点,除0下标以外的其他位置存放的是指向代表孙八的下标,这个0处下标存的是集合的所有元素的个数,且存放的是负数形式,这样存有一个好处,我们可以由这个并查集中有多少负数,从而判断这个并查集中有多少个集合。
  • 两个人产生关联,本质上是两个宿舍(集合)之间产生了关联,那两个宿舍如何进行关联起来呢?
  • 下面我们以图的形式更为清晰的进行表述:
    在这里插入图片描述
  • 也就是说因为宿舍的成员是以宿舍长联系起来的,那宿舍与宿舍之间,产生关联(合并),就宿舍长之间认识一下,两个集合就间接的关联起来了。
  • 下图是具体的存储方式:
    在这里插入图片描述

2.基本实现

 根据上面的描述,我们可以作出大致总结:

  1. 数组进行存储表示树形结构。
  2. 数组的下标对应着具体的信息(人名,编号等)。
  3. 我们可以通过一个元素的下标的值不断往上查找,直到找到找到小于0的,即为根节点所在的位置。
  4. 数组中负数的个数代表着集合的个数。
  5. 判断两个元素是否在同一个集合,只需找到根的下标判断是否相等即可。
  6. 将两个不同集合进行合并,其实就是找到根,然后进行更改一个根的指向与改变另一个根的元素个数即可。

由以上信息我们先可以搭建出实现并查集的大致框架:

2.1.基本框架

#include<iostream>
#include<vector>
#include<map>
using namespace std;
template<class T>
class UnionFindSet
{
public:
	UnionFindSet(const T* arr, size_t size);//构造函数
	
	int GetValueIndex(const T& val);//获取val所代表的下标。
	
	void GetRoot(const T& val);
	//获取根节点的下标
	
	void Union(const T& x1, const T& x2);
	//将两个元素的集合进行合并。
	
	bool IsSameSet(const T& x1, const T& x2);
	//判断两个元素是否在同一个集合中
	
	int GetSetSize(); 
	//获取集合的元素
private:
	map<T, int> _indexHash;
	//map或者unordered_map都可以。用于快速将T转换为对应的下标。
	
	vector<T> _createIndex;
	//用此数组对T类型元素生成下标。
	
	vetor<int> _aggregate; 
	//用于存放集合元素,即森林。
};

2.2.构造函数

	UnionFindSet(const T* arr, size_t size)
	{
		_aggreagte.resize(size, -1);
		//对存放集合的元素初始化,表示每个元素存放一个元素(负数表示)。
		
		_createIndex.resize(size);
		for (size_t i = 0; i < size; i++)
		{
			_createIndex[i] = arr[i];
			_indexHash[arr[i]] = i;//生成下标。
		}
	}

2.3.转换元素为下标

	int GetValueIndex(const T& val)
	{
		auto it = _indexHash.find(val);
		//最好判断一下val是否存在对应的下标。
		if (it == _indexHash.end())
		{
			throw invalid_argument("不存在所对应的下标");
			return -1;
		}
		return it->second;
	}

2.4.获取元素根节点下标

	int GetRoot(const T& val)
	{
		int index = GetValueIndex(val);

		//找不到小于0的下标指向的位置就一直向上进行找。
		while (_aggregate[index] >= 0)
		{
			index = _aggregate[index];
		}
		
		return index;
	}

2.5.判断元素集合是否相同

	bool IsSameSet(const T& x1, const T& x2)/
	{
		int index1 = GetRoot(x1);
		int index2 = GetRoot(x2);
		
		return index1 == index2;
	}

2.6.合并元素集合

	void Union(const T& x1, const T& x2)//将两个元素的集合进行合并。
	{
		if (!IsSameSet(x1, x2))
		{
			//不在同一个集合再进行合并。
			int index1 = GetRoot(x1);
			int index2 = GetRoot(x2);

			//进行一步优化,即元素少的合并到元素多的集合当中

			//此处我们假设index1为元素多的集合,index2为元素少的集合。
			if (abs(index1) < abs(index2))
			{
				swap(index1, index2);
			}
			//即将index2(少)合并到index1(多)上

			//将index2的元素加到index2上
			_aggregate[index1] += _aggregate[index2];

			//将index2的父路径指向index1
			_aggregate[index2] = index1;
		}
	}

2.7.获取集合个数

	int GetSetSize()//获取并查集的集合个数
	{
		int sum = 0;
		for (auto e : _aggregate)
		{
			//计算小于0的元素个数即可。
			if (e < 0)
			{
				sum++;
			}
		}
		return sum;
	}

3.路径压缩

 所谓路径压缩,其实解决存在这样的集合:
在这里插入图片描述
所引发的问题:如果数据足够的多,我们之前写的GetRoot函数的效率会急剧的降低,因此才需要路径压缩帮助我们进行优化。

实现方式也很简单:
在这里插入图片描述

  • 我们只需要找到根节点之后,再找一遍,此时将cur路径上的结点链接到root即可,这样方便了后续的查找。

  • 优化之后的GetRoot

	int GetRoot(const T& val)//获取根节点的下标
	{
		int index = GetValueIndex(val);

		int root = index;
		//找不到小于0的下标指向的位置就一直向上进行找。
		while (_aggregate[root] >= 0)
		{
			root = _aggregate[root];
		}
		
		//路径压缩进行优化。
		while (index != root)
		{
			//先保存之前父路径的下标
			int parent = _aggregate[index];
			//再将当前结点的父路径改为root
			_aggregate[index] = root;
			//继续往上迭代
			index = parent;
		}

		return root;
	}

4.源码与测试

  • UnionFindSet.hpp
#include<iostream>
#include<vector>
#include<map>
using namespace std;
template<class T>
class UnionFindSet
{
public:
	UnionFindSet(const T* arr, size_t size)
	{
		_aggregate.resize(size, -1);
		//对存放集合的元素初始化,表示每个元素存放一个元素(负数表示)。
		_createIndex.resize(size);
		for (size_t i = 0; i < size; i++)
		{
			_createIndex[i] = arr[i];
			_indexHash[arr[i]] = i;//生成下标。
		}
	}
	int GetValueIndex(const T& val)//获取val所代表的下标。
	{
		auto it = _indexHash.find(val);
		if (it == _indexHash.end())
		{
			throw invalid_argument("不存在所对应的下标");
			return -1;
		}
		return it->second;
	}

	int GetRoot(const T& val)//获取根节点的下标
	{
		int index = GetValueIndex(val);

		int root = index;
		//找不到小于0的下标指向的位置就一直向上进行找。
		while (_aggregate[root] >= 0)
		{
			root = _aggregate[root];
		}
		
		//路径压缩进行优化。
		while (index != root)
		{
			//先保存之前父路径的下标
			int parent = _aggregate[index];
			//再将当前结点的父路径改为root
			_aggregate[index] = root;
			//继续往上迭代
			index = parent;
		}

		return root;
	}
	void Union(const T& x1, const T& x2)//将两个元素的集合进行合并。
	{
		if (!IsSameSet(x1, x2))
		{
			//不在同一个集合再进行合并。
			int index1 = GetRoot(x1);
			int index2 = GetRoot(x2);

			//进行一步优化,即元素少的合并到元素多的集合当中

			//此处我们假设index1为元素多的集合,index2为元素少的集合。
			if (abs(index1) < abs(index2))
			{
				swap(index1, index2);
			}
			//即将index2(少)合并到index1(多)上

			//将index2的元素加到index2上
			_aggregate[index1] += _aggregate[index2];

			//将index2的父路径指向index1
			_aggregate[index2] = index1;
		}
	}
	
	//判断两个元素是否在同一个集合中
	bool IsSameSet(const T& x1, const T& x2)
	{
		int index1 = GetRoot(x1);
		int index2 = GetRoot(x2);
		
		return index1 == index2;
	}
	int GetSetSize()//获取并查集的集合个数
	{
		int sum = 0;
		for (auto e : _aggregate)
		{
			if (e < 0)
			{
				sum++;
			}
		}
		return sum;
	}
private:
	map<T, int> _indexHash;
	//map或者unordered_map都可以,用于快速将T转换为对应的下标。
	
	vector<T> _createIndex;//用此数组对T类型元素生成下标。
	
	vector<int> _aggregate; //用于存放集合元素,即森林。
};
  • Test.cpp
#include"UnionFindSet.hpp"
int main()
{
	string str[] = { "张三","李四","王五","赵六","周七" };
	UnionFindSet<string> ufs(str, sizeof(str) / sizeof(str[0]));
	ufs.Union("张三", "李四");
	ufs.Union("王五", "赵六");
	cout << "集合数为:" << ufs.GetSetSize() << endl;
	return 0;
}

运行结果:
在这里插入图片描述

并查集习题:

  1. 省份数量
  2. .等式方程的可满足性
  • 补充一下:
  1. 直接用下标进行抽象,是最常用的,因此这里的生成下标的vector与快速索引的map可以省去,形成一个简化版的并查集,更方便我们使用。
  2. 这里我们将并查集与图论放在一起,是因为并查集可以帮助起到判环的作用,因此我们这里放到一块进行讲解。

二、图论

1.基本概念

  • 图的概念有点凌乱,博主以思维导图的形式呈现出:

在这里插入图片描述

2.存储结构

  • 图有两个基本元素:
  1. 顶点, 我们可以将具体的顶点抽象成下标,从而用下标进行表示。
  2. 边,两个顶点即可确定一条边,因此我们可以用二维矩阵的方式进行表示;每个顶点都有与其相连的边,因此,我们可以单独每个顶点所连接的边抽象成桶的形式(类似于哈希桶)进行表示。
  • 因此我们通常有邻接矩阵和邻接表的形式进行存储。

2.1邻接矩阵

  • 实现代码:
	/*
	V(vertex) 表示实际存储边的类型,W(weight)表示边的权重,
	W_MAX 表示权重的不可能取值。
	Direction false表示是无向的,true表示是有向的。
	*/
	template<class V, class W, W W_MAX = INT_MAX, 
	bool Direction = false>
	class Graph
	{
	public:
		/*
		构造函数,传入的参数为V类型的指针指向的是V类型数组,
		以及数组的元素个数。
		*/
		Graph(const V* a, size_t n)//有多少个顶点
		{
			//初始化边,以及生成边的下标
			_vertexs.resize(n);
			for (size_t i = 0; i < n; i++)
			{
				_vertexs[i] = a[i];
				_indexMap[a[i]] = i;
			}
			//将矩阵进行初始化
			_matrices.resize(n);
			for (size_t i = 0; i < n; i++)
			{
				//没有权值,我们初始化为W_MAX,表示最开始顶点之间不互相连通。
				_matrices[i].resize(n, W_MAX);
			}
		}
		//将实际的顶点转换为对应的下标
		int GetVertexIndex(const V& v)
		{
			auto it = _indexMap.find(v);

			if (it == _indexMap.end())
			{
				//找不到
				throw invalid_argument("顶点不存在");//抛出异常
				return -1;
			}
			return it->second;
		}
		//添加边
		void AddEdge(const V& src, const V& dst, const W& w)
		{
			int srci = GetVertexIndex(src);
			int dsti = GetVertexIndex(dst);
			_AddEdge(srci, dsti, w);
		}
		//这里我们写一个子函数,方便内部接口进行使用。
		void _AddEdge(int srci, int dsti, const W& w)
		{
			_matrices[srci][dsti] = w;
			if (Direction == false)
			{
				//说明是无向图
				_matrices[dsti][srci] = w;
			}
		}
		//为了方便进行测试,这里博主将打印函数给出。
		void Print()
		{
			for (size_t i = 0; i < _vertexs.size(); i++)
			{
				printf("[%d]->", i);
				cout << _vertexs[i] << endl;
				//下标对应的边
			}
			cout << "    ";
			for (size_t i = 0; i < _matrices.size(); i++)
				printf("%-4d", i);
			cout << endl;
			for (size_t i = 0; i < _matrices.size(); i++)
			{
				printf("%-4d",i);
				for (size_t j = 0; j < _matrices[i].size(); j++)
				{
					if (_matrices[i][j] != W_MAX)
						printf("%-4d", _matrices[i][j]);
					else
						printf("%-4c", '*');
				}
				cout << endl;
			}
			cout << endl;
		}
		vector<V> _vertexs;//顶点
		map<V, int> _indexMap;//顶点所对应的下标
		vector<vector<W>> _matrices; //矩阵的英文
	};
  • 说明:
  1. 如果边带有权值,并且两个节点之间是连通的,边的关系就用权值代替。
  2. 如果两个顶点不通,则使用无穷大代替,即W_MAX。
  • 测试用例:

void TestGraph()
{
	Graph<char, int, INT_MAX, true> g("0123", 4);
	g.AddEdge('0', '1', 1);
	g.AddEdge('0', '3', 4);
	g.AddEdge('1', '3', 2);
	g.AddEdge('1', '2', 9);
	g.AddEdge('2', '3', 8);
	g.AddEdge('2', '1', 5);
	g.AddEdge('2', '0', 3);
	g.AddEdge('3', '2', 6);
	g.Print();
}
int main()
{
	TestGraph();
	return 0;
}

运行结果:
在这里插入图片描述

2.2邻接表

  • 实现代码:
namespace link
{
	/*
	因为要存顶点与边的关系,因此我们需要一个结构体来保存对应
	的相连的顶点与边的权值。
	*/
	template<class V,class W>
	struct Edge
	{
		V _dst;//目标顶点
		W _w;//权值
		Edge<V, W>* _next;
		//构造函数
		Edge(const V& dst, const W w)
			:_dst(dst),_w(w),_next(nullptr)
		{}
	};
	template<class V, class W, bool Direction = false>
	class Graph
	{
	public:
		typedef Edge<V, W> Edge;
		Graph(const V* a, size_t n)//有多少个顶点
		{
			//初始化边,以及生成对应的下标
			_vertexs.resize(n);
			for (size_t i = 0; i < n; i++)
			{
				_vertexs[i] = a[i];
				_indexMap[a[i]] = i;
			}
			//将矩阵进行初始化,为空表示最开始顶点没有边与之相连。
			_link.resize(n,nullptr);
			
		}
		//添加边
		void AddEdge(const V& src, const V& dst, const W& w)
		{
			int srci = GetVertexIndex(src);
			int dsti = GetVertexIndex(dst);
			Edge* node = new Edge(dst, w);
			node->_next = _link[srci];
			_link[srci] = node;
			if (Direction == false)
			{
				//说明是无向图
				Edge* node = new Edge(src, w);
				node->_next = _link[dsti];
				_link[dsti] = node;
			}
		}
		//获取顶点的下标。
		int GetVertexIndex(const V& v)
		{
			auto it = _indexMap.find(v);

			if (it == _indexMap.end())
			{
				//找不到
				throw invalid_argument("顶点不存在");//抛出异常
				return -1;
			}
			return it->second;
		}
		//打印的时候我们按照链表的形式打印即可。
		void Print()
		{
			for (size_t i = 0; i < _link.size(); i++)
			{
				cout << "[" << i << ":" << _vertexs[i] << "]->";
				Edge* cur = _link[i];
				while (cur)
				{
					cout << "[" << cur->_dst << ":" 
						<< _indexMap[cur->_dst] << ":" 
						<< cur->_w << "]->";
					
					cur = cur->_next;
				}
				cout << "nullptr" << endl;
			}
			cout << endl;
		}

	private:
		vector<V> _vertexs;//顶点
		map<V, int> _indexMap;//顶点所对应的下标
		vector<Edge*> _link; //邻接表

	};
}
  • 测试用例:
void TestGraph()
{

	string a[] = { "张三", "李四", "王五", "赵六" };
	Graph<string, int,true> g1(a, 4);
	g1.AddEdge("张三", "李四", 100);
	g1.AddEdge("张三", "王五", 200);
	g1.AddEdge("王五", "赵六", 30);
	g1.Print();
}

运行结果:

在这里插入图片描述


  • 总结:
  1. 邻接矩阵适合快速查看两个顶点的关系与路径权值。而对于顶点连接的边有多少,是什么,则需要遍历矩阵所在行进行确认。
  2. 邻接表适合直接取所有与点相连的边,而不适合快速查看两个顶点的关系。
  3. 因此邻接矩阵和邻接表是相辅相成的,而综合来看的话,对于较为稀疏的图,即顶点相连的边较少,平分秋色,各有千秋,而对于稠密的完全图来说,邻接矩阵更为合适。因此我们下面统一采用临界矩阵的方式进行实现。

3.遍历方式

3.1广度优先遍历

  • 图解:
    在这里插入图片描述

我们再来分析一下流程,这里是以A为起点,进行广度遍历。

  1. 先遍历A,。
  2. 然后遍历与A相连的BCD。
  3. 其次在遍历与BCD相连的EF,此时就需要注意之前访问过的结点不能在接着继续访问了。
  4. 接着遍历与EF相连的HG,此时也需注意同样的问题。
  5. 最后遍历与H相连的I,此时同理。
  • 因此广度优先遍历,需注意访问的时候不能再访问已经访问过的结点,其次访问时越访问越深的。

实现方式:

  1. 采用队列的结构,不断入与队列元素相连的未访问的结点。
  2. 使用一个vector 记录结点是否已经被访问过了,当入队列时,即将对应的结点的下标标记为true。
void BFS(const V& src)
{
	int srci = GetVertexIndex(src);
	int n = _vertexs.size();
	vector<int> is_visited(n, false);
	//防止重复结点入队列,以免形成回路。
	
	queue<int> que;
	
	que.push(srci);
	is_visited[srci] = true;
	int levelsize = 1;//第一层就srci.
	while (!que.empty())
	{
	
		for (int i = 0; i < levelsize; i++)
		{
			int front = que.front();
			que.pop();
			cout << front << ":" << _vertexs[front] << " ";
			//将与front相关的边进行入队列
			for (int i = 0; i < n; i++)
			{
				if (_matrices[front][i] != W_MAX &&
					is_visited[i] == false)
				{
					que.push(i);
					is_visited[i] = true;
				}
			}
			//这一层for循环式暴力遍历矩阵的所在行,确认是否有
			//没被访问的边。如果是邻接表就直接取较为方便,不过
			//稠密图倒是矩阵更优一点,能更好的确认两点的关系。					
		}
		cout << endl;
		//更新层结点的个数。
		levelsize = que.size();
	}
}
  • 测试用例:
	void TestBFS()
	{

		string a[] = { "A", "B", "C", "D", "E","F","G","H","I" };
		Graph<string, int> g1(a, sizeof(a) / sizeof(string));
		g1.AddEdge("A", "B", 1);
		g1.AddEdge("A", "C", 1);
		g1.AddEdge("A", "D", 1);

		g1.AddEdge("B", "E", 1);
		g1.AddEdge("B", "C", 1);

		g1.AddEdge("C", "F", 1);
		g1.AddEdge("C", "B", 1);

		g1.AddEdge("D", "F", 1);
		g1.AddEdge("E", "G", 1);
		g1.AddEdge("F", "H", 1);

		g1.AddEdge("H", "I", 1);


		g1.BFS("A");
	}
  • 运行结果:
    在这里插入图片描述

3.2深度优先遍历

  • 图解:
    在这里插入图片描述

我们再来分析一下流程,这里是以A为起点,进行深度遍历。

说明:已经访问过的结点我们是不再进行访问的。

  1. 先访问A相邻的B, 再访问与B相连的C, 再访问与C相连的F, 再访问与F相连的D。
  2. D相邻的A我们是不再进行访问的,因此又回到F, 接着访问H,紧接着访问与H相连的I,I没有访问过的结点,回退到H, H也没有访问过的结点回退到 F。
  3. F也没有与未访问的结点,回退到C,C也没有未访问的结点,于是回退到B。
  4. 接着访问与B相连的E, 更深一步访问与E相连的G,G没有未访问过的结点,回退到E, E此时也没有未访问过的结点回退到B, B此时也没有未访问过的结点,回退到A.
  5. 访问结束。
  • 实现代码:
	void _DFS(int srci,vector<bool>& is_visted)
	{
		for (size_t i = 0; i < is_visted.size(); i++)
		{
			if (_matrices[srci][i] != W_MAX && 
			is_visted[i] == false)
			{
			 	//此处打印的目的是便于测试。
				cout << "[" << _vertexs[srci] << "->" 
				<< _vertexs[i] << "]" << endl;
				is_visted[i] = true;
				_DFS(i, is_visted);
			}
		}
	}
	void DFS(const V& src)
	{
		int srci = GetVertexIndex(src);
		vector<bool> is_visted(_vertexs.size(), false);
		is_visted[srci] = true;
		_DFS(srci,is_visted);
	}
  • 测试用例:
void TestDFS()
{

	string a[] = { "A", "B", "C", "D", "E","F","G","H","I" };
	Graph<string, int> g1(a, sizeof(a) / sizeof(string));
	g1.AddEdge("A", "B", 1);
	g1.AddEdge("A", "C", 1);
	g1.AddEdge("A", "D", 1);

	g1.AddEdge("B", "E", 1);
	g1.AddEdge("B", "C", 1);

	g1.AddEdge("C", "F", 1);
	g1.AddEdge("C", "B", 1);

	g1.AddEdge("D", "F", 1);
	g1.AddEdge("E", "G", 1);
	g1.AddEdge("F", "H", 1);

	g1.AddEdge("H", "I", 1);

	g1.DFS("A");
}

/*主函数就自由发挥吧。*/
  • 运行结果:
    在这里插入图片描述

4.最小生成树

先来熟悉一下概念:

  • 最小生成树:图的生成树的路径最小。
  • 生成树:一个连通图的最小连通子图称作该图的生成树。有n个顶点的连通图的最小连通子图有n个顶点和n-1条边。
  • 连通图:若从顶点v1到顶点v2有路径,则称顶点v1与顶点v2是连通的。如果图中任意一对顶点都是连通的,则称此图为连通图。
  • 注意:连通图是无向图的概念,也就是说最小生成树的图必须是无向的。强连通图才是有向图的定义。

 简单的说就是从由n个顶点组成的连通图中选择n-1条边,子图连通且所边的权值相加最小。

  实现方法下面介绍克鲁斯卡尔和普里姆两种算法。

4.1Kruskal算法

  • 原理
  1. 首先将所有的边管理起来,每次取出最小的边。
  2. 判断已经选出的边是否构环,如果构成就弃置再从中选最小的边。
  3. (n个顶点构成的图)选择n-1条边即可。
  • 实现关键
  1. 用优先级队列对边进行管理。
  2. 用并查集进行判环。
  • 实现代码:
/*
为方便读者进行阅读,此处博主贴了一份并查集的简略代码。
*/
	template<class T>
	class UnionFindSet
	{
	public:
		//初始化大小,以及赋初值
		UnionFindSet(size_t size)
			:_pPath(size, -1)
		{}
		//将两个数进行合并
		void Union(int x1, int x2)
		{
			//找两个数的父结点
			int index1 = find(x1);
			int index2 = find(x2);
	
			//如果相同则说明已经在同一个集合下,无需进行合并
			if (index1 == index2) return;
	
			//将小的和在大的身上(优化防止路径过长)
			if (_pPath[index1] < _pPath[index2])
			{
				swap(index1, index2);
				swap(x1, x2);
			}
			//此处保证index1的父节点的数量多,index2的数量小
			_pPath[index1] += _pPath[index2];
			_pPath[index2] = index1;
	
		}
		//找根
		int GetValueIndex(int x)
		{
			//第一步:转换为下标
			int index = x;
			//第二步:根据下标找父节点
			while (_pPath[index] >= 0)
			{
				index = _pPath[index];
			}
			//找到父路径进行返回。
			//路径压缩
			while (x != index)
			{
				int parent = _pPath[x];
				_pPath[x] = index;
				x = parent;
			}
			return index;
		}
		int setsize()
		{
			int n = 0;
			for (int e : _pPath)
				if (e < 0) n++;
			return n;
		}
	private:
		vector<int> _pPath;
	};
	/*
	此结构体用于存放边的信息,放入优先级队列中便于进行管理。
	*/
	template<class W>
	struct Edge
	{
		int _srci;
		int _dsti;
		W _w;
		Edge(const int srci, const int dsti, const W& w)
			:_srci(srci), _dsti(dsti), _w(w)
		{}
		bool operator >(const Edge e) const
		{
			return _w > e._w;
		}
	};
	W Kruskal(self& min)
	{
		min._vertexs = _vertexs;
		//第一步,用优先级队列存放所有的边
		priority_queue<Edge, vector<Edge>, greater<Edge>> minque;
		size_t n = _vertexs.size();
		//无向图,只需存放一半的图的信息即可。
		for (size_t i = 0; i < n; i++)
		{
			for (size_t j = 0; j < i; j++)
			{
				if (_matrices[i][j] != W_MAX)
				{

					minque.push(Edge(i, j, _matrices[i][j]));
				}
			}
		}
		//第二步,选边,最小生成树,选择的边为 n-1条边
		size_t size = 0;
		UnionFindSet<int> u(n);
		W total = W();
		while (!minque.empty() && size != n-1)
		{
			Edge top = minque.top();
			minque.pop();
			if (u.find(top._dsti) != u.find(top._srci))
			{
				//说明不构成环,选择此边,并将其加入到并查集和表中

				//此处是为了方便测试。
				cout << _vertexs[top._dsti] << "->" 
				<< _vertexs[top._srci]<< ":" << top._w << endl;
				u.Union(top._dsti, top._srci);
				min._AddEdge(top._dsti, top._srci, top._w);
				size++;
				total += top._w;
			}
		}
		//队列为空跳出循环,因此需要判断一下看是否选出了n-1条边。
		if (size != n - 1)
		{
			//表明不能选出来
			return W();
		}
		return total;
	}
  • 测试用例:
	void TestGraphMinTree()
	{
		const char* str = "abcdefghi";
		Graph<char, int> g(str, strlen(str));
		g.AddEdge('a', 'b', 4);
		g.AddEdge('a', 'h', 8);
		//g.AddEdge('a', 'h', 9);
		g.AddEdge('b', 'c', 8);
		g.AddEdge('b', 'h', 11);
		g.AddEdge('c', 'i', 2);
		g.AddEdge('c', 'f', 4);
		g.AddEdge('c', 'd', 7);
		g.AddEdge('d', 'f', 14);
		g.AddEdge('d', 'e', 9);
		g.AddEdge('e', 'f', 10);
		g.AddEdge('f', 'g', 2);
		g.AddEdge('g', 'h', 1);
		g.AddEdge('g', 'i', 6);
		g.AddEdge('h', 'i', 7);
		Graph<char, int> kminTree(strlen(str));
		cout << "Kruskal:" << g.Kruskal(kminTree) << endl;
	}
/*main函数自由发挥吧*/
  • 运行结果:
    在这里插入图片描述

  • 图解:

在这里插入图片描述
说明:

  1. 程序走出的过程可能不一样,比如相同的边谁先选可能由优先级的实现原理决定,但大概率结果是一样的。
  2. 我们走出的只是局部的最优解,全局的最优解,可能还与相同的边的选择顺序有关,相同的边的如果互相影响,则可能会影响后面更大的边的选择。
  3. 因此如果所有的边互不相同那我们可以断定,此算法走出的最小生成树是确定的,即为全局的最小生成树。

4.2Prim算法

  • 原理
  1. 将顶点分为两个集合,设一个集合为X, 一个集合为Y。
  2. 选择一个起始点,放入X集合,剩余的顶点放入Y集合。
  3. 每次选择从Y中选择与X相连的最小的边,并将其相连的顶点放入X集合,从Y中丢弃此顶点。
  4. 直到选择 n - 1条边为止。
  • 实现关键:
  1. 将顶点分为两个集合X, Y,其实就避开了环的问题,产生环的原因本质就是一个集合内的两个顶点连到一块了。
  2. 我们选的是与集合X相连的最小的边,因此还要把X相连的边,放入优先级队列,往后循环可能会有一个集合内的边,我们只需判断边所连的目标顶点不在集合X即可,对于在集合X的我们不选即可。
  3. 除此之外,我们还需要确立一个起始点,用来初始化集合X和集合Y。
  • 实现代码:
	W Prim(self& min,const V& src)
	{
		size_t n = _vertexs.size();
		min._vertexs = _vertexs;
		/*
		第一步:选择顶点,作为起始顶点。分为两个数组,一个为起始数组
		,一个为选边数组
		*/
		int srci = GetVertexIndex(src);
		vector<bool> X(n,false);
		vector<bool> Y(n,true);
		X[srci] = true;
		Y[srci] = false;
		//第二步:将与srci相关的边入队列中。
		priority_queue<Edge, vector<Edge>, greater<Edge>> minque;
		for (size_t i = 0; i < n; i++)
		{
			//将边进行入队列
			if (_matrices[srci][i] != W_MAX)
			{
				minque.push(Edge(srci, i, _matrices[srci][i]));
			}
		}
		//第三步进行选边
		W total = W();
		size_t size = 0;
		while (!minque.empty())
		{
			Edge front = minque.top();
			minque.pop();
			//判断边的终点是否在X中
			if (X[front._dsti])
			{
				//说明构成环。
				cout << "构成环:";
				cout << _vertexs[front._srci] << "->" 
				<< _vertexs[front._dsti] << endl;
			}
			else
			{
				cout << _vertexs[front._srci] << "->" 
				<< _vertexs[front._dsti] << endl;
				++size;
				total += front._w;
				//将边添加到最小生成树里面,并将与dsti相连的边入队列
				min._AddEdge(front._srci, front._dsti, front._w);
				//将desi所在的集合进行删除与添加
				Y[front._dsti] = false;
				X[front._dsti] = true;
				//将dsti所连的边进行入队列
				for (size_t i = 0; i < n; i++)
				{
					//避免将已经入过的边再进行入队列
					if (_matrices[front._dsti][i] != W_MAX 
					&& Y[i])
					{
					//不在X[i] 即将在Y[i]进行入队列。
						minque.push(Edge(front._dsti, i,
						 _matrices[front._dsti][i]));
					}
				}
			}
		}

		//如果不能生成最小生成树。
		if (size != n - 1)
		{
			return W();
		}

		return total;
	}
  • 测试代码:
	void TestGraphMinTree()
	{
		const char* str = "abcdefghi";
		Graph<char, int> g(str, strlen(str));
		g.AddEdge('a', 'b', 4);
		g.AddEdge('a', 'h', 8);
		//g.AddEdge('a', 'h', 9);
		g.AddEdge('b', 'c', 8);
		g.AddEdge('b', 'h', 11);
		g.AddEdge('c', 'i', 2);
		g.AddEdge('c', 'f', 4);
		g.AddEdge('c', 'd', 7);
		g.AddEdge('d', 'f', 14);
		g.AddEdge('d', 'e', 9);
		g.AddEdge('e', 'f', 10);
		g.AddEdge('f', 'g', 2);
		g.AddEdge('g', 'h', 1);
		g.AddEdge('g', 'i', 6);
		g.AddEdge('h', 'i', 7);
		Graph<char, int> pminTree(strlen(str));
		cout << "Prim:" << g.Prim(pminTree, 'a') << endl;
		pminTree.Print();
	}
/*main 函数只需调用此函数即可*/

运行结果:

在这里插入图片描述

  • 图解:
    在这里插入图片描述

5.最短路径

  • 最短路径是描述两个顶点能连通的情况下,考虑两个顶点之间所经过路径的权值之和的最小值。
  • 举个例子,在现实世界中我们已经不关心两个地方能不能到的问题了,我们主要关系的是两个地方如何规划路程最短或者花费最低,诸如此类的问题,抽象到计算机即转换为了两个顶点所经过的路径的权值之和如何才能最短。

由此,我们引出迪杰斯特拉(Dijkstra), 贝尔曼福特(Bellman-Ford), 弗洛伊德(floyd warshall) 三种算法。

5.1Dijkstra算法

  • 基本认识
  • 此算法主要求的是不带负权值最小路径。

  • 算法思想主要在单源最短路径中进行体现。

  • 算法原理(贪心)
  1. 确定一个起始点,更新与其直接相连的顶点的路径。
  2. 选择路径和最短的那一个,此处确定了第一条路径最短的边。
  • 确定两字我们此处再稍作解释,由于已经选择了起始点直接到路径最短的顶点。因此不可能再出现,从起始点到另一个顶点再经过其它顶点到此点的路径和更短,更简单的表述是两点直接连着已经最短的了,再通过其它点绕远路只会更长,不会更短。
  • 此处用数学的语言进行描述或许更加直观。
  1. 再由最短的那个顶点,再更新(如果更小再进行更新)与其直接相连的边,再确定一条路径最短的边的顶点。由此顶点再进行更新。
  2. 如此往复,直到没有顶点可以更新,就结束。
  • 实现代码:
	void Dijkstra(const V& src, vector<W>& dst, vector<int>& pPath)
	{
		
		//将边与路径进行初始化
		size_t n = _vertexs.size();
		int srci = GetVertexIndex(src);
		//值初始化为W_MAX
		dst.resize(n, W_MAX);
		//路径初始化为-1
		pPath.resize(n, -1);
		//src->src路径值初始化为W(),路径初始化为srci
		dst[srci] = W();
		pPath[srci] = srci;

		//创建一个bool的vector使得每个结点只访问一次
		vector<bool> is_visted(n, false);
		for (size_t i = 0; i < n; i++)
		{
			W min = W_MAX;
			int vertexi = 0;
			//先选出没被访问过的最小的边
			for (size_t j = 0; j < n; j++)
			{
				if (!is_visted[j] && dst[j] < min)
				{
					min = dst[j];
					vertexi = j;
				}
			}
			//选出之后标记为选过的边
			is_visted[vertexi] = true;
			//再进行松弛更新与其相连的边
			for (size_t j = 0; j < n; j++)
			{
				/*
				首先得有边,且是顶点没有访问的点,并且 
				srci->vertex + vertex->j < srci->j,再进行更新
				*/			
				if (_matrices[vertexi][j] != W_MAX && !is_visted[j]
				&& dst[vertexi] + _matrices[vertexi][j] < dst[j])
				{
					//更新j的父路径和srci->j的距离
					pPath[j] = vertexi;
					dst[j] = dst[vertexi] + _matrices[vertexi][j];
				}
			}
		}
	}
  • 此处对这里的pPath进行说明一下,是将路径进行压缩从二维降到了一维,但其实也很简单,本质与并查集的路径表示大致一样,下标存的是父节点的下标。
  • 另外,这里打印时因为每个结点表示的是父结点的下标,因此我们还需将路径倒着找到之后,再翻转成正向的,再进行打印。
  • 打印最短路径函数:
void PrinrtShotPath(const V& src, vector<W>& dst, vector<int>& pPath)
{
	int srci = GetVertexIndex(src);
	size_t n = _vertexs.size();
	//先找到路径再进行逆置
	for (size_t i = 0; i < n; i++)
	{
		//不能是srci,要不然就陷入环了。
		if (i != srci)
		{
			vector<int> path;
			int parent = i;
			while (parent != srci)
			{
				path.push_back(parent);
				parent = pPath[parent];
			}
			//最后将srci根结点入进去
			path.push_back(srci);

			//逆转path得到路径
			reverse(path.begin(), path.end());
			for (auto index : path)
			{
				cout << _vertexs[index] << "->";
			}
			//最后打印出路径值
			cout << "最短路径值为:" << dst[i] << endl;
		}
	}
}
  • 测试用例:
	void TestGraphDijkstra()
	{
		const char* str = "syztx";
		Graph<char, int, INT_MAX, true> g(str, strlen(str));
		g.AddEdge('s', 't', 10);
		g.AddEdge('s', 'y', 5);
		g.AddEdge('y', 't', 3);
		g.AddEdge('y', 'x', 9);
		g.AddEdge('y', 'z', 2);
		g.AddEdge('z', 's', 7);
		g.AddEdge('z', 'x', 6);
		g.AddEdge('t', 'y', 2);
		g.AddEdge('t', 'x', 1);
		g.AddEdge('x', 'z', 4);
		vector<int> dist;
		vector<int> parentPath;
		g.Dijkstra('s', dist, parentPath);
		g.PrinrtShotPath('s', dist, parentPath);
	}
  • 运行结果:
    在这里插入图片描述

  • 图解:
    在这里插入图片描述

5.2Bellman-Ford算法

  • 用处:单源最短路径的负权值(不带负权回路)的图

  • 思想:暴力枚举遍历

  1. 由于只会更新出更短的路径,我们可以采取暴力枚举的方法。
  2. 将所有的边进行遍历,之后再遍历 n - 1 次进行修正。
  • 重点就在于: 为什么再遍历n - 1次 ?我们先来讨论一下,假设你再某次更新s->x->t->z 之后,s->x->t 出现了更短的路径(存在负权值,就有可能),更新成了s->y->t,但是原来已经更新的s->x->t->z虽然路径随着s->y->t更新,但是其s->t的权值并没有进行更新,这就导致了数据对不上的问题,因此我们需要再进行更新一轮,使之数据一致。而再次更新,有可能会导致其它最短路径的权值对不上,因此还要再进行更新,直到所有的最短路径都对上为止,因此最多要n-1次,带上最开始的那一次,总共n次。
  • 实现代码:
bool BellmanFord(const V& src, vector<W>& dst, vector<int>& pPath)
{
	//将边与路径进行初始化
	size_t n = _vertexs.size();
	int srci = GetVertexIndex(src);
	//值初始化为W_MAX
	dst.resize(n, W_MAX);
	//路径初始化为-1
	pPath.resize(n, -1);
	//src->src路径值初始化为W(),路径初始化为srci
	dst[srci] = W();
	pPath[srci] = srci;

	for (size_t k = 0; k < n; k++)
	{
		//更新n轮,因为一个路径更新出更短的路径,会影响其它路径的权值,
		//因此需要再次更新。
		//一轮之后,更新出最短路径,则其它路径的权值需要暴力更新一遍。
		//不带第一轮,最多更新n-1轮->其中每一轮都更新出了最短路径。
		bool update = false;
		for (size_t i = 0; i < n; i++)
		{
			for (size_t j = 0; j < n; j++)
			{
				//边存在,并且 s->i + i->j < s->j 
				if (_matrices[i][j] != W_MAX 
				&& dst[i] + _matrices[i][j] < dst[j])
				{
					update = true;
					//更新父路径和权值
					pPath[j] = i;
					dst[j] = dst[i] + _matrices[i][j];
				}
			}
		}

		if (!update)
		{
			break;
		}
	}

	//检查负权回路
	//再次更新一轮,检查是否能更新,如果还能更新,则存在负权回路。
	//如果没有更新,则为false,即
	bool is_existed = false;
	for (size_t i = 0; i < n; i++)
	{
		for (size_t j = 0; j < n; j++)
		{
			//边存在,并且 s->i + i->j < s->j 
			if (_matrices[i][j] != W_MAX 
			&& dst[i] + _matrices[i][j] < dst[j])
			{
				is_existed = true;
			}
		}
	}
	if (is_existed)
	{
		return false;
	}

	return true;
}
  • 测试用例:
	void TestGraphBellmanFord()
	{
		const char* str = "syztx";
		Graph<char, int, INT_MAX, true> g(str, strlen(str));
		g.AddEdge('s', 't', 6);
		g.AddEdge('s', 'y', 7);
		g.AddEdge('y', 'z', 9);
		g.AddEdge('y', 'x', -3);
		g.AddEdge('z', 's', 2);
		g.AddEdge('z', 'x', 7);
		g.AddEdge('t', 'x', 5);
		g.AddEdge('t', 'y', 8);
		g.AddEdge('t', 'z', -4);
		g.AddEdge('x', 't', -2);
		vector<int> dist;
		vector<int> parentPath;
		if (g.BellmanFord('s', dist, parentPath))
		{
			g.Print();
			g.PrinrtShotPath('s', dist, parentPath);
		}
		else
		{
			cout << "存在负权回路" << endl;
		}
	}
  • 运行结果:

在这里插入图片描述

  • 图解:

在这里插入图片描述

  • 说明:暴力更新,调试着看数据的变化效果更好。

  • 测试用例2:

	void TestGraphBellmanFord()
	{
		// 微调图结构,带有负权回路的测试
		const char* str = "syztx";
		Graph<char, int, INT_MAX, true> g(str, strlen(str));
		g.AddEdge('s', 't', 6);
		g.AddEdge('s', 'y', 7);
		g.AddEdge('y', 'x', -3);
		g.AddEdge('y', 'z', 9);
		g.AddEdge('y', 'x', -3);
		g.AddEdge('z', 's', -2);//更改此处见效更明显。
		g.AddEdge('z', 'x', 7);
		g.AddEdge('t', 'x', 5);
		g.AddEdge('t', 'y', 8);
		g.AddEdge('t', 'z', -4);
		g.AddEdge('x', 't', -2);
		vector<int> dist;
		vector<int> parentPath;
		if (g.BellmanFord('s', dist, parentPath))
		{
		 g.PrinrtShotPath('s', dist, parentPath);
		}
		else
		{
		 cout << "存在负权回路" << endl;
		}
	}
  • 运行结果:
    在这里插入图片描述
  • 图解:
    在这里插入图片描述
    说明:暴力循环完之后,再更新一次又会引起其它变小,此种情况只会越更新越小,求不出最小路径!

5.3floyd warshall算法

  • 用处:多源最短路径的负权值(不带负权回路)的图

  • 算法思想(dp):

  1. 拆分子问题:分为两种情况
  1. 所有的边经过点K.
  2. 所有的边不经过点K.
  3. 这里的K可能是所有的顶点。
  4. 因此求前两种情况的所有情况的最小值即可。

图解:
在这里插入图片描述

  • 实现代码:
void FloydWarshall(vector<vector<W>>& vvdst, 
vector<vector<int>>& vvpPath)
{
	size_t n = _vertexs.size();
	//初始化dst与pPath
	vvdst.resize(n);
	vvpPath.resize(n);
	for (size_t i = 0; i < n; i++)
	{
		vvdst[i].resize(n, W_MAX);
		vvpPath[i].resize(n, -1);
	}
	//再对边进行初始化,即将i直接到j的边先放在des数组中
	for (size_t i = 0; i < n; i++)
	{
		for (size_t j = 0; j < n; j++)
		{
			if (_matrices[i][j] != W_MAX)
			{
				vvdst[i][j] = _matrices[i][j];
				vvpPath[i][j] = i;
			}
			if (i == j)
			{
				//与此同时由于是距离,所以i == j  即 i->i 的距离为0
				vvdst[i][j] = 0;
			}
		}
	}

	for (size_t k = 0; k < n; k++)
	{
		//其中暴力选择k做为中间的边,分析是选择还是不选
		for (size_t i = 0; i < n; i++)
		{
			//从中进行选则两端的边
			for (size_t j = 0; j < n; j++)
			{
				//选择k作为中间的边,如果i->k,k->j < i->j
				//即分析是取k小还是不取k小,这里的k采用暴力枚举的方式。
				if (vvdst[i][k] != W_MAX && vvdst[k][j] != W_MAX
					&& vvdst[i][k] + vvdst[k][j] < vvdst[i][j])
				{
					//则需要更新dst[i][j]的父路径以及权值
					vvdst[i][j] = vvdst[i][k] + vvdst[k][j];
				/*
					i->k 更新 k->j,应为pPath[k][j]
					如果k->j中间没有其他结点,则说明 pPath[k][j] == k
					如果k->……->x->j中间经过了其它结点,则 pPath[k][j]==x
				*/
					vvpPath[i][j] = vvpPath[k][j];
					
				}
			}
		}
	}



	//此处我们打印出权值和路径的矩阵

	cout << "   ";
	for (size_t i = 0; i < n; i++)
	{
		printf("%-3d", i);
	}
	cout << endl;
	//1.权值矩阵
	for (size_t i = 0; i < n; i++)
	{
		printf("%-3d", i);
		for (size_t j = 0; j < n; j++)
		{
			if (vvdst[i][j] == W_MAX)
			{
				printf("%-3c", '*');
			}
			else
			{
				printf("%-3d", vvdst[i][j]);
			}
		}
		cout << endl;
	}
	printf("=============================================\n");
	//2.路径矩阵
	cout << "  ";
	for (size_t i = 0; i < n; i++)
	{
		cout << i << " ";
	}
	cout << endl;
	for (size_t i = 0; i < n; i++)
	{
		cout << i << " ";
		for (size_t j = 0; j < n; j++)
		{
			cout << vvpPath[i][j] << " ";
		}
		cout << endl;
	}
}
  • 测试用例:
	void TestFloydWarShall()
	{
		const char* str = "12345";
		Graph<char, int, INT_MAX, true> g(str, strlen(str));
		g.AddEdge('1', '2', 3);
		g.AddEdge('1', '3', 8);
		g.AddEdge('1', '5', -4);
		g.AddEdge('2', '4', 1);
		g.AddEdge('2', '5', 7);
		g.AddEdge('3', '2', 4);
		g.AddEdge('4', '1', 2);
		g.AddEdge('4', '3', -5);
		g.AddEdge('5', '4', 6);
		vector<vector<int>> vvDist;
		vector<vector<int>> vvParentPath;
		g.FloydWarshall(vvDist, vvParentPath);
		// 打印任意两点之间的最短路径
		for (size_t i = 0; i < strlen(str); ++i)
		{
			g.PrinrtShotPath(str[i], vvDist[i], vvParentPath[i]);
			cout << endl;
		}
	}

运行结果:
在这里插入图片描述

  • 图解:
    在这里插入图片描述

说明:这里II的矩阵表示的数字是真实下标对应的数字,我们这里打印的父路径的矩阵表示的数字是下标,因此还需要对不为-1的数加上1才对的上。

总结

  1. 并查集的原理和基本实现。
  2. 图论的基本概念,存储结构(邻接表和邻接矩阵),遍历方式(广度优先和深度优先),最小生成树的两个算法,最短路径的三个算法。
  • 并查集是一个较为简单的数据结构,而图论的表示形式是较为抽象的,需要我们将实际的例子抽象处理,因此不太好理解,关键在于多调试,多画图

尾序

我是舜华,期待与你的下一次相遇!

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

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

相关文章

JAVA:利用JUnit进行高效的单元测试

1、简述 在软件开发中&#xff0c;单元测试是确保代码质量和可维护性的关键步骤。JUnit作为Java领域最流行的单元测试框架之一&#xff0c;提供了简单而强大的测试工具&#xff0c;可以帮助开发者在项目开发过程中及时发现和修复代码中的问题。本文将介绍JUnit的基本用法以及一…

《Redis实战》学习笔记

特点 &#xff1a;1、是一个高性能的key/value内存型数据库 2、支持丰富的数据类型(string,List,Set,ZSet,Hash) 3、支持持久化 内存数据&#xff0c; 可以持久化到硬盘中 4、单进程&#xff0c;单线程 效率高 redis实现分布式锁 一、redis的相关指令 1、flushDB 清空当前…

一起学量化之ATR指标

1. ATR指标定义 ATR,全称Average True Range,即均幅指标,是一个衡量市场波动性的技术分析工具。它通过计算一定周期内的平均真实波幅,来反映市场波动的程度。ATR指标由J. Welles Wilder Jr开发,主要用于研判买卖时机和管理交易风险。下图中蓝色的线即为ATR。 2. ATR指标应…

【PTA-C语言】实验七-函数与指针I

如果代码存在问题&#xff0c;麻烦大家指正 ~ ~有帮助麻烦点个赞 ~ ~ 目录——实验七-函数与指针I 6-1 弹球距离&#xff08;分数 10&#xff09;6-2 使用函数输出一个整数的逆序数&#xff08;分数 10&#xff09;6-3 使用函数求最大公约数&#xff08;分数 10&#xff09;6-4…

Wi-Fi 7用到的一些技术术语汇总

文章目录 low-complexity ap coordinationMulti-AP coordinationHARQ&#xff08;Hybrid Automatic Repeat Request&#xff09;multiple resource units per STAFCC open 6GhzU-NII-5,U-NII-6&#xff1f;4096-QAM&#xff08;Quadrature Amplitude Modulation&#xff0c;正交…

在FC中手工创建虚拟机模板

1、Linux去除个性化信息 &#xff08;1&#xff09;编辑网卡配置文件&#xff0c;只保留以下内容&#xff08;以RHEL 7为例&#xff09; &#xff08;2&#xff09;清除主机密钥信息&#xff08;开机会自动生成&#xff09; &#xff08;3&#xff09;清除Machine ID&#xff…

数据统计的一些专业术语学习

数据统计的一些专业术语学习 1. 极差2. 方差3. 标准差4. 均值绝对差 1. 极差 数据统计的极差&#xff0c;又称全距&#xff0c;是指一组数据中最大值和最小值之差。 举个例子&#xff0c;如果我们有一组数据&#xff1a;1&#xff0c;2&#xff0c;3&#xff0c;4&#xff0c…

【动态规划】【字符串】C++算法:正则表达式匹配

作者推荐 视频算法专题 涉及知识点 动态规划 字符串 LeetCode10:正则表达式匹配 给你一个字符串 s 和一个字符规律 p&#xff0c;请你来实现一个支持 ‘.’ 和 ‘’ 的正则表达式匹配。 ‘.’ 匹配任意单个字符 ’ 匹配零个或多个前面的那一个元素 所谓匹配&#xff0c;是…

【SD】一致性角色 - 同一人物 不同姿势 - 2

首先生成4张不同姿势的图片 masterpiece,high quality,(white background:1.6),(simple background:1.4),1gril,solo,black footwear,black hair,brown eyes,closed mouth,full body,glasses,jacket,long hair,long sleeves,lookig at viewer,plaid,plaid skirt,pleated shirt,…

人是需要被肯定和认可的,赞美也是一种动力

前几天转发了一些网上的文章&#xff0c;突然有了10个关注我的人&#xff0c;赞美数和收藏量也上去了一些。 这是一种很意外的惊喜。 看了一下主题是&#xff1a; 1,如何将.NET8创建的控制台程序部署成WINDOWS服务。 2,.NET进阶篇06-async异步、thread多线程 3,易语言启动线程传…

病理HE学习贴(自备)

目录 正常结构 癌症HE 在线学习 以胃癌的学习为例 正常结构 1&#xff1a;胃粘膜正常结构和细胞分化 ●表面覆盖小凹上皮细胞(主要标志物&#xff1a;MUC5AC)以保护黏膜。 ●胃底腺固有腺体由黏液颈细胞(MUC6)、主细胞(Pepsinogen l)和壁细胞(Proton pump α-subunit)组…

css动态传参,attr的妙用

今天再做一个编辑器的功能的时候&#xff0c;发现有一段非常奇妙的代码&#xff0c;使用attr获取div标签的data-label值。 css的attr?What fuck?这又是什么鬼东西&#xff0c;emmm。 查询后官方是这么回答的&#xff1a; CSS 表达式 attr() 用来获取选择到的元素的某一 HTM…

Windows 10启用Hyper-V

Windows 10启用Hyper-V 官网教程PowerShell 启用 Hyper-V启用 Hyper-V 角色 我们知道VMware是创建虚拟机的好工具&#xff0c;那Windows平台上有没有虚拟工具呢&#xff1f; 今天我们要讲解的就是Windows才入局的虚拟工具&#xff1a;Hyper-V 官网教程 https://learn.microsof…

VMware安装RHEL9.0版本Linux系统

最近在学习Linux&#xff0c;安装了Red Hat Enterprise Linux 的 9.0版本&#xff0c;简称RHEL9.0。RHEL9.0是Red Hat公司发布的面向企业用户的Linux操作系统的最新版本。我把它安装在虚拟机VMware里来减少电脑性能占用&#xff0c;也防止系统炸搞得我后面要重装。安装RHEL9.0还…

vu3-14

第一个需求是在用户登录成功之后&#xff0c;在主页显示用户的真实姓名和性别&#xff0c;这些信息要调用后端API获取数据库里面的信息&#xff0c;第二个需求是点击菜单1&#xff0c;在表单中修改用户信息之后&#xff0c;更新到后端数据库&#xff0c;然后在主页同步更新用户…

24. 二维数组

二维数组的创建和初始化 #include<stdio.h> int main() {//二维数组的创建int arr[3][4]; //三行四列char ch[3][10]; //三行十列return 0; } 创建&#xff1a; 完全初始化 int arr[3][4] { 1,2,3,4,5,6,7,8,9,10,11,12 }; 不完全初始化 int arr1[3][4] { 1,2,3…

【兔子王赠书第14期】《YOLO目标检测》涵盖众多目标检测框架,附赠源代码和全书彩图!

文章目录 写在前面YOLO目标检测推荐图书本书特色内容简介作者简介 推荐理由粉丝福利写在后面 写在前面 小伙伴们好久不见吖&#xff0c;本期博主给大家推荐一本关于YOLO目标检测的图书&#xff0c;该书侧重目标检测的基础知识&#xff0c;包含丰富的实践内容&#xff0c;是目标…

微服务篇之Nacos快速入门

Nacos 简介 Nacos 起源 Nacos 起源于阿里巴巴 2008 年的五彩石项目&#xff08;完成微服务拆分和业务中台建设&#xff09;&#xff0c;经历了阿里十年双十⼀的洪峰流量的考验&#xff0c;沉淀了简单易用、稳定可靠、性能卓越等核心特性。随着云计算的兴起和受到开源软件行业…

Java项目调试实战:如何高效调试Spring Boot项目中的GET请求,并通过equalsIgnoreCase()解决大小写不一致问题

Java项目调试实战&#xff1a;如何高效调试Spring Boot项目中的GET请求&#xff0c;并通过equalsIgnoreCase解决大小写不一致问题 写在最前面全部过程Java equalsIgnoreCase() 方法idea中如何调试SpringBoot项目在IntelliJ IDEA中使用内置HTTP客户端设置断点和调试 补充&#x…

【阅读笔记】Semi-supervised Domain Adaptation in Graph Transfer Learning

Background 真实世界的图上节点的标签数据是很难拿到的。 因此图转移学习被提出将知识从标记的源图转移出来&#xff0c;以帮助预测域变化的目标图中节点的标签。 尽管图迁移学习算法取得了重大进展&#xff0c;但它们通常假定源图中的所有节点都被标记出来了。 因此文章定义…