算法学习系列(十四):并查集

目录

  • 引言
  • 一、并查集概念
  • 二、并查集模板
  • 三、例题
    • 1.合并集合
    • 2.连通块中点的数量

引言

这个并查集以代码短小并且精悍的特点,在算法竞赛和面试中特别容易出,对于面试而言,肯定不会让你去写一两百行的代码,一般出的都是那种比较短的,而且还不好想考验思维的那种题,那并查集就将这两点全占了,所以重要性很大,而且竞赛的话也就是将多个知识点合并起来考察,这个也很可能成为一个点,所以话不多说就开始吧。

一、并查集概念

  • 并查集主要有两个作用:
    1.查询两个元素是否在同一集合中,时间复杂度近乎O(1)
    2.将两个集合合并

  • 初始化思路:首先有多个元素,它们最初每个集合只有它们自己,有个p[N]数组,p[i]代表 i 号结点的父结点的下标,p [i] = i

  • 合并思路:先查询它们各自的父结点a,b,然后让p [a] = b 意为a的父亲为b这样a所在的集合就与b所在的集合合并了,如下图所示
    在这里插入图片描述

  • 查询思路:查询自己的父结点是不是根结点,如果是返回,不是则继续递归再次查找父结点的父结点,最后返回根结点,这个是通过递归实现的,这里有个路径压缩的过程,就是最后一个是根结点就会返回,那么就把根结点赋给倒数第三层、第四层…,也就是让每一个结点都指向根结点,类似就是下图所示
    在这里插入图片描述

二、并查集模板

void init()
{for(int i = 1; i <= n; ++i) p[i] = i;
}int find(int x)  //查询编号为x的父结点
{if(p[x] != x) p[x] = find(p[x]);  //说明不是父结点,让当前结点指向父结点的父结点,也就是最终的根结点return p[x];  //返回父结点
}

三、例题

1.合并集合

一共有 n 个数,编号是 1∼n,最开始每个数各自在一个集合中。现在要进行 m 个操作,操作共有两种:
M a b,将编号为 a 和 b 的两个数所在的集合合并,如果两个数已经在同一个集合中,则忽略这个操作;
Q a b,询问编号为 a 和 b 的两个数是否在同一个集合中;输入格式
第一行输入整数 n 和 m。
接下来 m 行,每行包含一个操作指令,指令为 M a b 或 Q a b 中的一种。输出格式
对于每个询问指令 Q a b,都要输出一个结果,如果 a 和 b 在同一集合内,则输出 Yes,否则输出 No。
每个结果占一行。数据范围
1≤n,m≤105输入样例:
4 5
M 1 2
M 3 4
Q 1 2
Q 1 3
Q 3 4输出样例:
Yes
No
Yes
#include <cstdio>
#include <cstring>
#include <iostream>using namespace std;const int N = 1e5+10;int p[N];  //p[i]代表i的父节点的编号
int n, m;  //n个结点,m次询问void init()
{for(int i = 1; i <= n; ++i) p[i] = i;
}int find(int x)  //查询编号为x的父结点
{if(p[x] != x) p[x] = find(p[x]);  //说明不是父结点,让当前结点指向父结点的父结点,也就是最终的根结点 ,注意这里不能写成find(p[x])不然条件就一直没有变化return p[x];  //返回父结点
}int main()
{scanf("%d%d", &n, &m);init();while(m--){char op[2];int a, b;scanf("%s%d%d", op, &a, &b);if(!strcmp(op, "M")){a = find(a), b = find(b);  //找到a和b的根结点p[a] = b;  //让a指向b,意为a的父亲为b,那么a下的所有结点的父亲都是b}else{if(find(a) == find(b)) printf("Yes\n");  //如果两个结点的根结点相同,则在同一个集合else printf("No\n");}}return 0;
}

所有的用例都通过了
在这里插入图片描述
在这里插入图片描述

2.连通块中点的数量

给定一个包含 n 个点(编号为 1∼n)的无向图,初始时图中没有边。现在要进行 m 个操作,操作共有三种:
C a b,在点 a 和点 b 之间连一条边,a 和 b 可能相等;
Q1 a b,询问点 a 和点 b 是否在同一个连通块中,a 和 b 可能相等;
Q2 a,询问点 a 所在连通块中点的数量;输入格式
第一行输入整数 n 和 m。
接下来 m 行,每行包含一个操作指令,指令为 C a b,Q1 a b 或 Q2 a 中的一种。输出格式
对于每个询问指令 Q1 a b,如果 a 和 b 在同一个连通块中,则输出 Yes,否则输出 No。
对于每个询问指令 Q2 a,输出一个整数表示点 a 所在连通块中点的数量每个结果占一行。数据范围
1≤n,m≤105输入样例:
5 5
C 1 2
Q1 1 2
Q2 1
C 2 5
Q2 5输出样例:
Yes
2
3
#include <cstdio>
#include <cstring>
#include <iostream>using namespace std;const int N = 1e5+10;int p[N], cnt[N];  //来判断根结点所对应的集合的数量,只有根结点有意义
int n, m;int find(int x)
{if(x != p[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;for(int i = 1; i <= n; ++i) cnt[i] = 1;while(m--){char op[5];int a, b;scanf("%s", op);if(!strcmp(op,"C")){scanf("%d%d", &a, &b);a = find(a), b = find(b);if(a == b) continue;p[a] = b;cnt[b] += cnt[a];}else if(!strcmp(op,"Q1")){scanf("%d%d", &a, &b);if(find(a) == find(b)) printf("Yes\n");else printf("No\n");}else{scanf("%d", &a);printf("%d\n", cnt[find(a)]);}}return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

是否在消息上打标?

场景 用户可以创建某个虚拟形象&#xff0c;将该形象发布到聊天平台&#xff0c;然后和该形象进行聊天。 用户创建虚拟形象的时候&#xff0c;开启备份功能&#xff0c;将自己的聊天记录备份到指定位置&#xff0c;也可以关闭开关&#xff0c;停止备份功能。 聊天功能和备份…

双指针——移动零

题目 示例 算法原理 我们使用两个指针&#xff0c;cur扫描数组&#xff0c;如果nums[cur]为非0&#xff0c;dest&#xff0c;然后让nums[cur]与nums[dest]交换&#xff0c;从而实区间[0,dest]为非0,[dest1,cur]为0&#xff0c;[cur,numsSize-1]为未扫描 题目链接&#xff1a;28…

网络安全—PKI公钥基础设施

文章目录 前提知识散列函数非对称加密数字签名 PKI受信任的人RA注册CA颁发IKE数字签名认证&#xff08;交换证书&#xff09;密钥管理 前提知识 散列函数 散列也可以叫哈希函数&#xff0c;MD5、SHA-1、SHA-2、、&#xff08;不管叫啥&#xff0c;都记得是同一个东西就行&…

uni-app设置地图显示

使用前需到**高德开放平台&#xff08;https://lbs.amap.com/&#xff09;**创建应用并申请Key 登录 高德开放平台&#xff0c;进入“控制台”&#xff0c;如果没有注册账号请先根据页面提示注册账号 打开 “应用管理” -> “我的应用”页面&#xff0c;点击“创建新应用”&…

ESP32入门六(读取引脚的模拟信号[4]:Arduino-ESP32 ADC API详解)

在之前的章节中&#xff0c;我们测试了读取引脚的模拟值&#xff0c;ADC功能在实际中用途十分广泛&#xff0c;在本章中&#xff0c;我们把一些常用的ADC函数做一个详细的说明。 ADC单次性模式 ADC单次模式API与Arduino的analogRead功能完全兼容。当您调用analogRead或analogR…

C语言——小细节和小知识7

一、逆序字符串 1、递归1 #include <stdio.h> #include <string.h>void ReverseArray(char *str) {char temp *str;//1int len (int)strlen(str);*str *(str len - 1);//2*(str len - 1) \0;//3if(strlen(str 1) > 2)//只要字符串还大于2&#xff0c;就…

【Spring实战】16 Profile

文章目录 1. 定义2. 使用2.1 定义 Profile2.2 激活 Profile 3. 演示3.1 properties文件3.2 打印日志3.3 启动服务&验证3.4 修改 active3.5 重启服务&验证 4. 应用场景4.1 数据库配置4.2 日志配置 5. 代码详细总结 Spring 框架提供了一种强大的机制&#xff0c;允许在不…

关于“Python”的核心知识点整理大全55

目录 注意 3. 模板 topic.html 4. 将显示所有主题的页面中的每个主题都设置为链接 topics.html 18.5 小结 第 19 章 用户账户 19.1 让用户能够输入数据 19.1.1 添加新主题 1. 用于添加主题的表单 forms.py 2. URL模式new_topic urls.py 3. 视图函数new_topic() …

51单片机中TCON, IE, PCON等寄存器的剖析

在单片机中&#xff0c;如何快速通过名字记忆IQ寄存器中每一个控制位的作用呢&#xff1f; IE&#xff08;interrupt enable&#xff09;寄存器中&#xff0c;都是中断的使能位置。 其中的EA&#xff08;enable all&#xff09;是总使能位&#xff0c;ES(enable serial)是串口…

excel统计分析——单因素方差分析

参考资料&#xff1a;生物统计学 方差分析是将总变异分解为组间变异的方差和组内变异的方差&#xff0c;并通过F检验推断处理效应是否显著的过程&#xff0c;而方差是通过平方和与自由度计算出来的&#xff0c;所以方差分析首先需要进行平方和与自由度的分解。具体步骤如下&…

简单FTP客户端软件开发——JavaFX开发FTP客户端

文章目录 导入外部包commons-net-3.10.0.jarJavaFX开发客户端 FTP客户端要求如下&#xff1a; 简单FTP客户端软件开发 网络环境中的一项基本应用就是将文件从一台计算机中复制到另一台可能相距很远的计算机中。而文件传送协议FTP是因特网上使用得最广泛的文件传送协议。FTP使用…

yolov5目标检测神经网络——损失函数计算原理

前面已经写了4篇关于yolov5的文章&#xff0c;链接如下&#xff1a; 1、基于libtorch的yolov5目标检测网络实现——COCO数据集json标签文件解析 2、基于libtorch的yolov5目标检测网络实现(2)——网络结构实现 3、基于libtorch的yolov5目标检测网络实现(3)——Kmeans聚类获取anc…