图论-并查集

**并查集(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;
}
相关推荐
对方正在长头发丿12 小时前
LETTERS(DFS)
c++·笔记·算法·深度优先·图论
WG_1718 小时前
第五章.图论
算法·图论
玉树临风ives1 天前
leetcode 2360 图中最长的环 题解
算法·leetcode·深度优先·图论
Joe_Wang52 天前
[图论]拓扑排序
数据结构·c++·算法·leetcode·图论·拓扑排序
蒙奇D索大2 天前
【数据结构】图解图论:度、路径、连通性,五大概念一网打尽
数据结构·考研·算法·图论·改行学it
君义_noip2 天前
信息学奥赛一本通 1524:旅游航道
c++·算法·图论·信息学奥赛
刃神太酷啦3 天前
基础算法篇(3)(蓝桥杯常考点)-图论
数据结构·c++·算法·职场和发展·蓝桥杯·图论·蓝桥杯c++组
羑悻的小杀马特3 天前
【狂热算法篇】探寻图论幽径:Bellman - Ford 算法的浪漫征程(通俗易懂版)
c++·算法·图论·bellman_ford算法
好好学习^按时吃饭4 天前
P4551 最长异或路径
算法·深度优先·图论
triticale4 天前
【图论】最短路径问题总结
java·开发语言·图论