算法刷题笔记 合并集合(C++实现)

文章目录

题目描述

  • 一共有n个数,编号是1∼n,最开始每个数各自在一个集合中。
  • 现在要进行m个操作,操作共有两种:
    • M a b,将编号为ab的两个数所在的集合合并,如果两个数已经在同一个集合中,则忽略这个操作;
    • Q a b,询问编号为a和b的两个数是否在同一个集合中;

输入格式

  • 第一行输入整数nm
  • 接下来m行,每行包含一个操作指令,指令为M a bQ a b中的一种。

输出格式

  • 对于每个询问指令Q a b,都要输出一个结果,如果ab在同一集合内,则输出Yes,否则输出No
  • 每个结果占一行。

数据范围

  • 1 ≤ n,m ≤ 10^5

基本思路

  • 并查集:并查集是本题使用的数据结构。这是一个常考的数据结构,其代码非常短,但是思路都很精巧。
  • 并查集的作用 :快速将两个集合合并,以及判定两个元素是否属于同一个集合。这里的快速是指时间复杂度近乎O(1)
  • 基本原理:以一棵树的形式(不一定是二叉树)来维护一个集合,每一个集合的编号就是该集合的根节点的编号。树的每一个结点都存储该结点的父结点的位置。每次查询某个元素属于哪一个集合,只需要找到该元素对应的结点,然后一级一级向上查找父结点直到根结点即可。尽管已经有一定的优化,但是一直查找到树的根节点也需要进行遍历,所以有进一步的优化,即找到一个集合树的根节点后,该路径上的所有结点的父节点都会直接指向根结点。这种优化方法被称为路径压缩。
  • 判定是否是根节点:一个结点的父节点编号和结点本身相同。可以基于这个条件来一级一级向上查找。
  • 如何合并两个集合:将一个集合的树作为另一个集合树的根节点的一个子结点即可。
  • 按秩合并:并查集的另一种优化方法,但是优化效果不显著,使用情况较少,此处不再赘述。
  • find(x):这是并查集最常用的方法之一,作用是返回元素x所在的集合的编号。

实现代码

cpp 复制代码
#include <iostream>
using namespace std;

// sets中的每一个元素表示当前集合的根节点编号
const int N = 100010;
int sets[N];

// 用于查找一个元素所属的集合编号的函数
int find_set(int x)
{
    // 对于非根结点,则查询其上方的一个结点,同时进行路径压缩
    if(sets[x] != x) sets[x] = find_set(sets[x]);
    else return sets[x];
}

// 合并两个集合的函数,将第二个元素所属的集合作为第一个元素所属集合的叶子结点
inline void merge(int a, int b)
{
    // 首先通过find_set找到b元素所在集合的编号
    // 然后将该编号对应的根节点元素(与编号同号)的所属集合设置为a元素所在集合的编号
    sets[find_set(b)] = find_set(a);
}

// 查询两个元素是否属于同一个集合的函数
inline void query(int a, int b)
{
    // 如果查询两个元素的集合编号相同,则属于同一个集合
    if(find_set(a) == find_set(b)) cout << "Yes" << endl;
    else cout << "No" << endl;
}

int main(void)
{
    int n, m;
    cin >> n >> m;
    // 初始化集合
    for(int i = 1; i <= n; ++ i) sets[i] = i;
    // 进行合并和查询操作
    for(int i = 0; i < m; ++ i) 
    {
        char operation;
        int a, b;
        cin >> operation >> a >> b;
        if(operation == 'M') merge(a, b);
        else if(operation == 'Q') query(a, b);
    }
    return 0;
}

注意事项

  • 本题代码中的核心是理解sets数组的含义。sets数组实际上存储的是每一个元素所属的集合编号,初始情况下每个元素的集合编号和自身的编号相同,即一一对应。
  • find_set函数中,如果当前元素不是某一个集合的根节点,那就将当前元素所属的集合更新为当前元素的集合的根节点所属的集合,即进行路径压缩。
相关推荐
查理零世2 分钟前
【蓝桥杯集训·每日一题2025】 AcWing 6134. 哞叫时间II python
python·算法·蓝桥杯
仟濹2 分钟前
【二分搜索 C/C++】洛谷 P1873 EKO / 砍树
c语言·c++·算法
紫雾凌寒11 分钟前
解锁机器学习核心算法|神经网络:AI 领域的 “超级引擎”
人工智能·python·神经网络·算法·机器学习·卷积神经网络
YH_DevJourney33 分钟前
Linux-C/C++《C/8、系统信息与系统资源》
linux·c语言·c++
Igallta_8136221 小时前
【小游戏】C++控制台版本俄罗斯轮盘赌
c语言·开发语言·c++·windows·游戏·游戏程序
MiyamiKK572 小时前
leetcode_位运算 190.颠倒二进制位
python·算法·leetcode
C137的本贾尼2 小时前
解决 LeetCode 串联所有单词的子串问题
算法·leetcode·c#
青橘MATLAB学习2 小时前
时间序列预测实战:指数平滑法详解与MATLAB实现
人工智能·算法·机器学习·matlab
lingllllove2 小时前
matlab二维艾里光束,阵列艾里光束,可改变光束直径以及距离
开发语言·算法·matlab
88号技师2 小时前
2025年2月一区SCI-海市蜃楼搜索优化算法Mirage search optimization-附Matlab免费代码
开发语言·人工智能·算法·机器学习·matlab·优化算法