图论-并查集

**并查集(Union-find Sets)**是一种非常精巧而实用的数据结构,它主要用于处理一些不相交集合的合并问题.一些常见的用途有求连通子图,求最小生成树Kruskal算法和最近公共祖先(LCA)等.

并查集的基本操作主要有:

.1.初始化

2.查询find

3.合并union

一般我们都会采用路径压缩 这样效率更加高

cpp 复制代码
#include<iostream>
#include<cstdio>
#include<cstdlib>
using namespace std;
#define MAXN 20001
int fa[MAXN];
void init(int n) {
	for (int i = 1; i <= n; i++) {
		fa[i] = i;
	}//初始化
}
int find(int x) {
	if (x == fa[x]) {
		return x;

	}
	else {
		fa[x] = find(fa[x]);//路径压缩 也就是一直找到祖先
		return fa[x];
	}
}
void unionn(int i, int j) {
	int i_fa = find(i);//找到i的祖先
	int j_fa = find(j);//找到j的祖先
	fa[i_fa] = j_fa;//i的祖先指向j的祖先 反过来也可以
}
int main() {
	int n, m, x, y, q;
	scanf("%d", &n);
	init(n);
	scanf("%d", &m);
	for (int i = 1; i <= m; i++) {
		scanf("%d%d", &x, &y);
		unionn(x, y);
	}
	scanf("%d", &q);
	for (int i = 1; i <= q; i++) {
		scanf("%d%d", &x, &y);
		if (find(x) == find(y)) {
			printf("Yes\n");
		}
		else {
			printf("No\n");
		}

	}


	return 0;
}

或者这样写

cpp 复制代码
#include<iostream>
#include<cstring>
#include<algorithm>

using namespace std;

const int N = 20010;

int n, m;
int p[N];
int find(int x) {
	if (p[x] != x)p[x] = find(p[x]);
	return p[x];
}
int main() {
	scanf("%d%d", &n, &m);
	for (int i = 1; i <= n; i++) p[i] = i;
	while (m--) {
		int a, b;
		scanf("%d%d", &a, &b);
		p[find(a)] = find(b);//合并 a->b
	}
	scanf("%d,&m");
	while (m--) {

		int a, b;
		scanf("%d%d", &a, &b);
		if (find(a) == find(b))puts("yes");
		else puts("no");
	}
	return 0;
	
}
cpp 复制代码
#include<iostream>
using namespace std;

const int N = 10010;

int n, m;
int p[N];

int find(int x) {
	if (p[x] != x)p[x] = find(p[x]);
	return p[x];
}
int main() {
	scanf("%d%d", &n, &m);
	for (int i = 1; i <= n; i++) p[i] = i;
	char op[2];//读入操作的字符串  因为字符串后面有'\0'所以要存多一位
	while (m--) {
		int a, b;
		scanf("%s%d%d",&op ,&a, &b);
		if(*op=='M')p[find(a)] = find(b);//合并
		else {
			if (find(a) == find(b)) {
				puts("Yes");
			}
			else {
				puts("No");
			}
		}

	}

	return 0;
}

cpp 复制代码
#include<iostream>
using namespace std;
const int N = 10010;

int n, m;
int p[N], s[N];

int find(int x) {
	if (p[x] != x)p[x] = find(p[x]);
	return p[x];
}
int main() {
	scanf("%d%d", &n, &m);
	for (int i = 1; i <= n; i++) p[i] = i, s[i] = 1;
	while (m--)
	{
		char op[3];
		int a, b;
		scanf("%s", &op);
		if (*op == 'C') {
			scanf("%d%d", &a, &b);
			a = find(a), b = find(b);
			if (a != b) {//如果相等证明他们在同一个祖先中
				s[b] += s[a];
				p[a] = b;
			}
			else if (*op == 'Q1') {
				scanf("%d%d", &a, &b);
				if (find(a) == find(b)) {
					puts("Yes\n");
				}
				else {
					puts("No\n");
				}
			}
			else {
				scanf("%d", &a);
				printf("%d\n", s[find(a)]);
			}
		}
	}
	return 0;
}
相关推荐
一只鱼^_2 天前
牛客周赛 Round 108
数据结构·c++·算法·动态规划·图论·广度优先·推荐算法
_Coin_-2 天前
算法训练营DAY58 第十一章:图论part08
数据结构·算法·图论
闪电麦坤955 天前
数据结构:图的表示 (Representation of Graphs)
数据结构·算法·图论
BlackPercy5 天前
【图论】Graphs.jl 最小生成树算法文档
算法·图论
SuperCandyXu6 天前
洛谷 P3128 [USACO15DEC] Max Flow P -普及+/提高
c++·算法·图论·洛谷
zc.ovo6 天前
牛子图论1(二分图+连通性)
数据结构·c++·算法·深度优先·图论
ltrbless6 天前
最小生成树算法详解
算法·排序算法·图论
love you joyfully7 天前
图论简介与图神经网络(Dijkstra算法,图卷积网络GCN实战)
人工智能·深度学习·神经网络·算法·贪心算法·图论
YA10JUN8 天前
数据结构基础--最小生成树
数据结构·算法·图论
啊我不会诶11 天前
【图论】最短路算法
算法·图论