图论-并查集

**并查集(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 天前
【拓扑排序】-- 算法原理讲解,及实现拓扑排序,附赠热门例题
java·算法·图论
elseif1233 天前
【C++】并查集&家谱树
开发语言·数据结构·c++·算法·图论
我爱工作&工作love我4 天前
P4913 【深基16.例3】二叉树深度 dfs-二叉树的遍历
算法·深度优先·图论
拼好饭和她皆失6 天前
图论:最小生成树,二分图详细模板及讲解
c++·算法·图论
颢珂智库Haokir Insights7 天前
离散数学 (Discrete Mathematics) 的数学模型示例:图论 (Graph Theory)
图论
仰泳的熊猫7 天前
题目1433:蓝桥杯2013年第四届真题-危险系数
数据结构·c++·算法·蓝桥杯·深度优先·图论
燃于AC之乐8 天前
《算法实战笔记》第10期:六大算法实战——枚举、贪心、并查集、Kruskal、双指针、区间DP
算法·贪心算法·图论·双指针·区间dp·二进制枚举
北温凉8 天前
EEG 有效连接分析(基于 MVGC 与图论)
图论·eeg·mvgc·脑功能连接
仰泳的熊猫8 天前
题目1432:蓝桥杯2013年第四届真题-剪格子
数据结构·c++·算法·蓝桥杯·深度优先·图论
一条大祥脚9 天前
ABC357 基环树dp|懒标记线段树
数据结构·算法·图论