【数据结构】并查集

并查集是简单的数据结构,学会并查集,为图打好基础。

并查集的概念

是树状的数据结构,用于处理相交集合的合并与查询

通常用森林表示,一片森林表示一个集合

并查集一般需要完成

  1. 查找元素属于哪个集合
  2. 查看两个元素是否属于同一个集合
  3. 将两个集合归并成一个集合
  4. 集合的个数


并查集的原理
 

假设有10个人,用集合表示为{0,1,2,3,4,5,6,7,8,9}

我们用数组表示这十个人

  1. 数组的下标表示人的编号
  2. 数组的内容表示每个人认识其中人的数目 (-1 表示只认识自己)

表示为10片森林

经过一段时间后,他们形成三个小团体,s1{0,6,7,8}  s2{1,4,9} s3{ 2,3,5}

利用并查集表示

  • 数组的下标对应集合中元素的编号
  • 数组中如果为负数,负号代表根,数字代表该集合中元素个数
  • 数组中如果为非负数,代表该元素双亲在数组中的下标

所以在并查集中,我们会关注数组的下标和数组的内容。

如果数组的内容是负数,那么他就是根(Root)

如果数组的内容为(非负数),那么就指向他的父亲。

所以我们能简单解决这些问题

  • 查找元素属于哪个集合

沿着树形关系一直往上寻找到根(直到找到内容为负数的)

  • 查看元素是否属于同一个集合

寻找到根,不同则表示不是同一个集合

  • 将两个集合归并成一个集合

一个集合归并到另一个集合中,并修改集合的内容

  • 集合的个数

多少个负数内容的位置,就有多少个集合

接下来,来简单实现一个并查集


并查集的模拟

框架

//并查集
class UnionFindSet {
public://构造函数UnionFindSet(int n);//查找元素所在的集合int findRoot(int x);//合并两个元素所在的集合void Union(int x1, int x2)//获取并查集中集合的个数int SetCount();
private:vector<int> _set;各个结点之间的关系
};

构造

接收vector容器应该预留的空间

初始化为-1

	UnionFindSet(int size):_set(size,-1){}

不需要自定析构函数

默认会调用系统默认析构


查找root

一直向上寻找,变换x ,直到找到内容为负数

	//查找元素是否在集合里size_t FindRoot(int x){while (_set[x] >= 0){x = _set[x];}return x;}

合并俩个集合

将B合并到A中,A的根内容要改变,B的根内容也要改变

	//合并void Union(int x1, int x2){int root1 = FindRoot(x1);int root2 = FindRoot(x2);if (root1 != root2){_set[root1] += _set[root2];_set[root2] = root1;	}}

路径压缩

如果有大量的数据,那么树的层数可能非常高,查找的时候就需要一层一层往上迭代。这时候很浪费效率,这里就提出路径压缩。

路径压缩一般发送在查找根结点,会压缩路径上的所有结点,挂接到根上。

	size_t FindRoot(int x){//找根size_t root = x;while (_set[x] >= 0){x = _set[x];}//压缩while (_set[x] >= 0){size_t parent = _set[x];_set[x] = root;x = parent;}return root;}

Gitee:提取代码

相关题目

LCR 116. 省份数量

思路:
运用并查集的相关知识

题目给我们一个相邻矩阵,遍历一半矩阵就能知道连通的关系。

我们把矩阵中为1的放到一个集合中,返回集合的数目

解题时,运用lambda表达式,调用找根root的函数。

class Solution {
public:int findCircleNum(vector<vector<int>>& isConnected) {int n=isConnected[0].size();vector<int> _ufs(n,-1);//找根 >=0 就往上找auto FindRoot=[&_ufs](int x){int parent=x;while(_ufs[parent]>=0){parent=_ufs[parent];}return parent;};//遍历,遇到1 并且根不同 就合并 for(int i=0;i<n;i++){for(int j=0;j<=i;j++){if(isConnected[i][j]==1){int root1=FindRoot(i);int root2=FindRoot(j);if(root1!=root2){_ufs[root1]+=_ufs[root2];_ufs[root2]=root1;}}}}int count =0;for(auto x:_ufs){if(x<0)count++;}return count;}
};

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

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

相关文章

C++二叉树进阶——二叉搜索树

二叉搜索树 1. 二叉树的概念2. 二叉树的实现2.1创建节点类2.2 查找Find2.3 插入Insert2.4 删除Erase2.5 中序遍历2.6 构造/析构 3. 递归实现3.1 查找FindR3.2 插入InsertR3.3 删除EraseR 4.整体代码 1. 二叉树的概念 二叉搜索树又称二叉排序树&#xff0c;它或者是一棵空树&am…

springboot743二手交易平台

springboot743二手交易平台 获取源码——》公主号&#xff1a;计算机专业毕设大全

2024 年 11 款最佳 iPhone 数据恢复软件和应用程序

数据丢失是任何人都无法承受的&#xff0c;因为它对每个人都至关重要。但导致数据丢失的原因有很多&#xff0c;一些常见的原因是意外删除数据、设备被盗、iOS 越狱、硬件损坏、病毒感染等。我们列出了 iOS 的顶级恢复工具&#xff0c;其中包括&#xff1a;将帮助您方便地恢复数…

相机图像质量研究(20)常见问题总结:CMOS期间对成像的影响--全局快门/卷帘快门

系列文章目录 相机图像质量研究(1)Camera成像流程介绍 相机图像质量研究(2)ISP专用平台调优介绍 相机图像质量研究(3)图像质量测试介绍 相机图像质量研究(4)常见问题总结&#xff1a;光学结构对成像的影响--焦距 相机图像质量研究(5)常见问题总结&#xff1a;光学结构对成…

【Java多线程】对进程与线程的理解

目录 1、进程/任务&#xff08;Process/Task&#xff09; 2、进程控制块抽象(PCB Process Control Block) 2.1、PCB重要属性 2.2、PCB中支持进程调度的一些属性 3、 内存分配 —— 内存管理&#xff08;Memory Manage&#xff09; 4、线程&#xff08;Thread&#xff09;…

Go语言的100个错误使用场景(40-47)|字符串函数方法

前言 大家好&#xff0c;这里是白泽。 《Go语言的100个错误以及如何避免》 是最近朋友推荐我阅读的书籍&#xff0c;我初步浏览之后&#xff0c;大为惊喜。就像这书中第一章的标题说到的&#xff1a;“Go: Simple to learn but hard to master”&#xff0c;整本书通过分析100…

如何在Windows中配置多个显示器?这里提供详细步骤

Windows可以通过多种方式使用多个显示器,扩展或复制主显示器。你甚至可以关闭主显示器。以下是如何使用简单的键盘快捷键更改辅助显示设置。 使用Windows+P投影菜单 要快速更改Windows 10处理多个显示器的方式,请按Windows+P。屏幕右侧会弹出一个名为“投影”的深灰色菜单。…

Base64编码的优点与缺点

title: Base64编码的优点与缺点 date: 2024/2/16 14:06:37 updated: 2024/2/16 14:06:37 tags: Base64编码ASCII转换数据传输文本存储安全性数据膨胀字符串解码 Base64编码是一种将二进制数据转换为可打印ASCII字符的编码方式。它被广泛应用于数据传输和存储&#xff0c;以提升…

- 工程实践 - 《QPS百万级的有状态服务实践》01 - 存储选型实践

本文属于专栏《构建工业级QPS百万级服务》 《QPS百万级的无状态服务实践》已经完成。截止目前为止&#xff0c;支持需求“给系统传入两个日期&#xff0c;计算间隔有多少天”的QPS百万级服务架构已经完成。如图1&#xff1a; 图1 可是这个架构不能满足需求“给系统传入两个日期…

node+vue3+mysql前后分离开发范式——实现对数据库表的增删改查

文章目录 ⭐前言⭐ 功能设计与实现💖 node后端操作数据库实现增删改查💖 vue3前端实现增删改查⭐ 效果⭐ 总结⭐ 结束⭐结束⭐前言 大家好,我是yma16,本文分享关于 node+vue3+mysql前后分离开发范式——实现对数据库表的增删改查。 技术选型 前端:vite+vue3+antd 后端:…

用HTML和CSS打造跨年烟花秀视觉盛宴

目录 一、程序代码 二、代码原理 三、运行效果 一、程序代码 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><title>跨年烟花秀</title><meta name"viewport" content"widthdevi…

【Linux】 Linux 小项目—— 进度条

进度条 基础知识1 \r && \n2 行缓冲区3 函数介绍 进度条实现版本 1代码实现运行效果 版本2 Thanks♪(&#xff65;ω&#xff65;)&#xff89;谢谢阅读&#xff01;&#xff01;&#xff01;下一篇文章见&#xff01;&#xff01;&#xff01; 基础知识 1 \r &&a…