【高阶数据结构】并查集 -- 详解

news/2024/9/22 18:21:12/

一、并查集的原理

1、并查集的本质和概念

(1)本质

并查集的本质:森林。


(2)概念

在一些应用问题中,需要将 n 个不同的元素划分成一些不相交的集合

开始时,每个元素自成一个单元素集合,然后按一定的规律将归于同一组元素的集合合并。在此过程中要反复用到查询某一个元素归属于那个集合的运算。适合于描述这类问题的抽象数据类型称为并查集(union-find set)


比如:某公司今年校招,在全国总共招生 10 人,广州招 4 人,深圳招 3 人,东莞招 3 人,10 个人来自不同的学校,起先互不相识,每个学生都是一个独立的小团体。

现给这些学生进行编号:{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}; 给以下数组用来存储该小集体,数组中的数字代表:该小集体中具有成员的个数。

(初始状态为 -1 ,表示 10 棵树,即每个学生是一个独立的集合)

(跟堆类似,用下标表示关系,双亲表示法)

毕业后,学生们要去公司上班,每个地方的学生自发组织成小分队一起上路,于是:广州学生小分队 s1={0,6,7,8},深圳学生小分队 s2={1,4,9},东莞学生小分队 s3={2,3,5} 就相互认识了,10 个人形成了三个小团体。

假设右三个群主 0, 1, 2 担任队长,负责大家的出行。

一趟火车之旅后,每个小分队成员就互相熟悉,称为了一个朋友圈。

从上图可以看出:
  • 编号 6, 7, 8 同学属于 0 号小分队,该小分队中有 4 人(包含队长 0);
  • 编号为 4和 9 的同学属于 1 号小分队,该小分队有 3 人(包含队长 1);
  • 编号为 3 和 5 的同学属于 2 号小分队,该小分队有 3 个人(包含队长 1)。
仔细观察数组中内融化,可以得出以下结论:
  1. 数组的下标对应集合中元素的编号
  2. 数组中如果为负数负号代表根数字的绝对值代表该集合中元素个数
  3. 数组中如果为非负数,代表该元素双亲在数组中的下标

2、并查集的应用

在公司工作一段时间后,广州小分队中 8 号同学与深圳小分队 1 号同学奇迹般的走到了一起,两个小圈子的学生相互介绍,最后成为了一个小圈子:

现在 0 集合有 7 个人,2 集合有 3 个人,总共两个朋友圈。

通过以上例子可知,并查集一般可以解决一下问题:
(1)查找元素属于哪个集合
沿着数组表示树形关系以上一直找到根(即:树中中元素为负数的位置)。

(2)查看两个元素是否属于同一个集合
沿着数组表示的树形关系往上一直找到树的根,如果根相同表明在同一个集合,否则不在。

(3)将两个集合归并成一个集合
将两个集合中的元素合并将一个集合名称改成另一个集合的名称。

(4)统计集合的个数
遍历数组,数组中元素为负数的个数即为集合的个数。

二、并查集的实现

class UnionFindSet
{
public:// 初始时,将数组中元素全部设置为1UnionFindSet(size_t n):_ufs(n, -1){}void Union(int x1, int x2){int root1 = FindRoot(x1);int root2 = FindRoot(x2);// 如果x1已经与x2在同一个集合就不需要合并了if (root1 == root2)return;if (root1 > root2)swap(root1, root2);// 将两个集合中元素合并_ufs[root1] += _ufs[root2];// 将其中一个集合名称改变成另外一个_ufs[root2] = root1;}// 给一个元素的编号,找到该元素所在集合的名称int FindRoot(int x){int parent = x;// 如果数组中存储的是负数,找到,否则一直继续while (_ufs[parent] >= 0){parent = _ufs[parent];}return parent;}bool InSet(int x1, int x2){return FindRoot(x1) == FindRoot(x2);}// 数组中负数的个数,即为集合的个数size_t SetSize(){size_t size = 0;for (size_t i = 0; i < _ufs.size(); i++){if (_ufs[i] < 0){size++;}}return size;}private:vector<int> _ufs;
};

http://www.ppmy.cn/news/1428550.html

相关文章

C++修炼之路之list--C++中的双向循环链表

目录 前言 一&#xff1a;正式之前先回顾数据结构中的双向循环链表 二&#xff1a;list的简介 三&#xff1a;STL中list常用接口函数的介绍及使用 1.构造函数接口 2.list迭代器 范围for 3.数据的修改接口函数 4.list容量操作函数 5.list的迭代器失效 6.演示代码和测…

iOS RACScheduler 使用详解

RACScheduler 是 ReactiveCocoa 框架中的一个关键组件&#xff0c;用于在 iOS 开发中管理任务的并发执行。以下是如何详细使用 RACScheduler 的指南&#xff0c;以 Markdown 格式展示。 主要调度器 主线程调度器 用于在主线程上执行任务&#xff0c;通常用于 UI 更新操作。 …

2024年4月13日美团春招实习试题【第一题:好子矩阵】-题目+题解+在线评测【模拟】

2024年4月13日美团春招实习试题【第一题:好子矩阵】-题目题解在线评测【模拟】 题目描述&#xff1a;输入描述输出描述样例 解题思路一&#xff1a;模拟解题思路二&#xff1a;思路二解题思路三&#xff1a;直接判断 题目描述&#xff1a; 塔子哥定义一个矩阵是”好矩阵”&…

四.RocketMQ的几种消息发送方式应用

RocketMQ的几种消息发送方式应用 一&#xff1a;普通消息1&#xff09;发送同步消息2&#xff09;发送异步消息3&#xff09;单向发送消息4&#xff09;消费消息-负载均衡模式5&#xff09;消费消息-广播模式 二&#xff1a;顺序消息1.顺序消息指的是:严格按照消息的发送顺序进…

ubuntu安装QEMU

qemu虚拟机的使用&#xff08;一&#xff09;——ubuntu20.4安装QEMU_ubuntu安装qemu-CSDN博客 遇到的问题&#xff1a; (1)本来使用git clone https://github.com/qemu/qemu.git fatal: 无法访问 https://github.com/qemu/qemu.git/&#xff1a;GnuTLS recv error (-110): …

OpenHarmony实战开发-页面深色模式适配。

介绍 本示例介绍在开发应用以适应深色模式时&#xff0c;对于深色和浅色模式的适配方案&#xff0c;采取了多种策略如下&#xff1a; 1. 固定属性适配&#xff1a;对于部分组件的颜色属性&#xff0c;如背景色或字体颜色&#xff0c;若保持不变&#xff0c;可直接设定固定色值…

SQL中WITH RECURSIVE的用法

SQL中WITH RECURSIVE的用法 文章目录 SQL中WITH RECURSIVE的用法定义**WITH RECURSIVE 结构通常包含以下几个关键部分&#xff1a;****1. CTE&#xff08;Common Table Expression&#xff0c;公用表表达式&#xff09;&#xff1a;**2.递归查询的结构3.连接操作符&#xff1a;…

【C语言】每日一题,快速提升(2)!

&#x1f525;博客主页&#x1f525;&#xff1a;【 坊钰_CSDN博客 】 欢迎各位点赞&#x1f44d;评论✍收藏⭐ 题目&#xff1a;杨氏矩阵 有一个数字矩阵&#xff0c;矩阵的每行从左到右是递增的&#xff0c;矩阵从上到下是递增的&#xff0c;请编写程序在这样的矩阵中查找某个…