经典算法思想--并查集
前言
(最近在学习Java,所有函数都是用Java语言来书写的)
- 前言部分是一些前提储备知识
在并查集(Union-Find)数据结构中,rank(中文称为“秩”)是用来表示树的高度或深度的一种辅助信息。它的主要作用是优化合并操作,以保持并查集的结构尽可能扁平,从而提高查询效率。
秩的具体定义
-
秩(Rank):
- 秩用来衡量一个节点在树中的相对高度。具体来说,秩通常指的是树的“深度”或“高度”。初始时,每一个节点的秩可以设定为 0。若秩突然增加,说明该节点的子树的深度在增加。
-
合并时使用:
- 在进行
union操作时,- 若两个集合的根节点的秩不同,我们将根节点秩更小的树连接到秩更大的树上。
- 当两个根节点的秩相同时,将任意一棵树连接到另一棵树上,并将新根节点的秩值加一。这可以避免树过高。
- 在进行
合并示例
- 初始状态:
- 每个元素最初是自己父节点(根节点),并且秩都是 0。
Element: 1 2 3 4
Parent: [1, 2, 3, 4] // 表示每个元素都是自己的父节点
Rank: [0, 0, 0, 0] // 秩初始化为 0
- 调用
union(1, 2):- 首先找出元素 1 和 2 的根节点。由于它们各自是自己的根节点,所以
find(1)返回 1,find(2)返回 2。 - 由于根节点不同(1 不是 2),可以将它们合并。
- 因为它们的秩相等,都为 0,所以可以任意选择一个作为新的根节点,此处选择把 2 的父节点设为 1,并将 1 的秩增加 1。
- 首先找出元素 1 和 2 的根节点。由于它们各自是自己的根节点,所以
Union(1, 2):
Element: 1 2
Parent: [1, 1] // 2 的父节点指向 1(1成为新的根)
Rank: [1, 0] // 将 1 的秩增加 1
- 状态更新后的图示:
1/2
- 接着进行
union(2, 3):- 查找根节点:
find(2)返回 1(2 的父节点是 1),find(3)返回 3。 - 1 和 3 是不同的根节点,可以合并。
- 由于 1 的秩(1)大于 3 的秩(0),所以将 3 的父节点指向 1。
- 查找根节点:
Union(2, 3):
Element: 1 2 3
Parent: [1, 1, 1] // 3 的父节点指向 1
Rank: [1, 0, 0] // 秩不变
- 最后进行
union(1, 3):- 查找根节点:
find(1)返回 1,find(3)返回 1,所以它们已经在同一个集合中,什么也不做。
- 查找根节点:
总结
在进行 union 操作时,我们首先需要找到两个元素的根节点。如果它们的根节点不同,就可以将它们合并。如果相同,则表示它们已经在同一个集合中。
以上是对 union 操作的正确描述和过程演示,谢谢你的耐心!如果你还有其他问题,请随时问我。
并查集简介
并查集(Union-Find)是一种用于处理不重叠集合的数据结构。它特别适合用于解决有关集合连接、合并和查询的问题。并查集通常包括两个主要操作:
- Find: 查找元素所属的集合。
- Union: 合并两个集合。
并查集的主要思想
- 快速查找:利用树形结构可以快速找到集合的根代表。
- 树的优化:
- 路径压缩:在查找过程中,将访问的每个节点直接连接到根节点,从而优化树的结构,使得树变平,查找效率更高。
- 按秩合并:在合并过程中,总是将秩(rank)低的树连接到秩高的树,防止树变得过于高。
路径压缩
路径压缩是在 find 操作中进行的优化。当我们执行查找时,我们将经过的所有节点直接连接到根节点。这减少了树的高度,从而提高随后的查找效率。
路径压缩示例代码
public int find(int[] parent, int index) {if (parent[index] != index) {// 递归地找到根节点,并在回溯阶段将当前节点直接连接到根节点parent[index] = find(parent, parent[index]);}return parent[index];
}
路径压缩示例说明
假设我们有初始集合 [1, 2, 3, 4, 5],其树结构如下:
1
|
2 - 3 - 4|5
- 假设
parent = [0, 1, 1, 2, 3, 3],这意味着 2 和 3 的父节点是 1,5 的父节点是 3。 - 调用
find(parent, 5)时,路径为[5 -> 3 -> 1]。 - 路径压缩将改变
5的父节点直接连接到1,结果是parent = [0, 1, 1, 1, 3, 1]。 - 树在调用 find 后,将如下一步变得更平展:
1
|
2 - 3 - 4 - 5
按秩合并
按秩合并是在 union 操作中进行的优化,目的是尽可能保持树的扁平。所谓“秩”,在这里可以理解为树的高度。
按秩合并示例代码
public void union(int[] parent, int[] rank, int index1, int index2) {int root1 = find(parent, index1);int root2 = find(parent, index2);if (root1 != root2) {if (rank[root1] > rank[root2]) {parent[root2] = root1; // root2合并到root1上} else if (rank[root1] < rank[root2]) {parent[root1] = root2; // root1合并到root2上} else {parent[root2] = root1; // 随意合并并增加其中一个的rankrank[root1]++;}}
}
按秩合并示例说明
假设我们有两个树:
- 第一棵树的根为 A,秩为 2。
- 第二棵树的根为 B,秩为 3。
调用 union(parent, rank, A, B) 时:
- 由于 B 的秩大于 A,A 被合并到 B 上,这样就避免了增加树的高度。
- 在 rank 相同的情况下,任选一个作为新根,并增加该树的 rank。
结合路径压缩和按秩合并
结合这两个优化策略,在大多数实际应用中,find 和 union 操作可以接近于常数时间复杂度。这种效率使得并查集在处理大量集合合并和查找操作时极为高效。使用上面的两个优化版本的代码,能够保证树的高度不会过于增长,从而优化操作效率。
并查集的应用
并查集被广泛应用于很多算法与实际问题中,比如:
并查集(Union-Find)数据结构在计算机科学中有着广泛的应用,特别是在处理图相关的问题时。下面我将介绍几个具体的实例问题,并展示如何使用并查集来解决这些问题。
1. 网络连接
问题:判断网络中两个节点是否连通。
实例:假设我们有一个网络系统,每一对节点之间有或没有直接连接。如果两个节点是连通的,则它们之间存在一条直接或间接路径。
代码:
public class Network {private int[] parent;private int[] rank;public Network(int size) {parent = new int[size];rank = new int[size];for (int i = 0; i < size; i++) {parent[i] = i;rank[i] = 0;}}public int find(int x) {if (parent[x] != x) {parent[x] = find(parent[x]);}return parent[x];}public void union(int x, int y) {int rootX = find(x);int rootY = find(y);if (rootX != rootY) {if (rank[rootX] > rank[rootY]) {parent[rootY] = rootX;} else if (rank[rootX] < rank[rootY]) {parent[rootX] = rootY;} else {parent[rootY] = rootX;rank[rootX]++;}}}public boolean isConnected(int x, int y) {return find(x) == find(y);}public static void main(String[] args) {Network network = new Network(5);network.union(0, 1);network.union(1, 2);System.out.println(network.isConnected(0, 2)); // 输出 trueSystem.out.println(network.isConnected(0, 3)); // 输出 false}
}
2. 图的连通分量
问题:找出图中的连接组件,即连通分量。
实例:给定一个无向图,找出所有的连通分量。
代码:
import java.util.*;public class Graph {private int[] parent;private int[] rank;public Graph(int size) {parent = new int[size];rank = new int[size];for (int i = 0; i < size; i++) {parent[i] = i; // 初始化,每个节点的父节点指向自己 rank[i] = 0; // 秩初始化为0 }}// 查找并进行路径压缩 public int find(int x) {if (parent[x] != x) {parent[x] = find(parent[x]);}return parent[x];}// 联合操作 public void union(int x, int y) {int rootX = find(x);int rootY = find(y);if (rootX != rootY) {if (rank[rootX] > rank[rootY]) {parent[rootY] = rootX;} else if (rank[rootX] < rank[rootY]) {parent[rootX] = rootY;} else {parent[rootY] = rootX;rank[rootX]++;}}}// 计算连通分量的数量 public int countComponents() {Set<Integer> uniqueRoots = new HashSet<>();for (int i = 0; i < parent.length; i++) {uniqueRoots.add(find(i));}return uniqueRoots.size();}// 主函数用于测试 public static void main(String[] args) {Graph graph = new Graph(5);graph.union(0, 1);graph.union(1, 2);graph.union(3, 4);System.out.println(graph.countComponents()); // 输出 2,表明有两个连通分量 }
}
3. 最小生成树算法中的环检测
问题:在 Kruskal 算法中,检测添加的边是否会形成环。
实例:找到一个连通无向图的最小生成树。
代码:
import java.util.*;class Edge implements Comparable<Edge> {int src, dest, weight;Edge(int src, int dest, int weight) {this.src = src;this.dest = dest;this.weight = weight;}@Overridepublic int compareTo(Edge compareEdge) {return this.weight - compareEdge.weight;}
}public class KruskalMST {private List<Edge> edges;private int vertices;public KruskalMST(int vertices) {this.vertices = vertices;edges = new ArrayList<>();}public void addEdge(int src, int dest, int weight) {edges.add(new Edge(src, dest, weight));}public List<Edge> findMST() {Collections.sort(edges);int[] parent = new int[vertices];int[] rank = new int[vertices];// Initialize parent and rank arrays for (int i = 0; i < vertices; i++) {parent[i] = i;rank[i] = 0;}List<Edge> mst = new ArrayList<>();// Traverse through all edges for (Edge edge : edges) {int rootSrc = find(parent, edge.src);int rootDest = find(parent, edge.dest);// Check if the selected edge forms a cycle if (rootSrc != rootDest) {mst.add(edge);union(parent, rank, rootSrc, rootDest);}}return mst;}private int find(int[] parent, int x) {if (parent[x] != x) {parent[x] = find(parent, parent[x]);}return parent[x];}private void union(int[] parent, int[] rank, int x, int y) {int rootX = find(parent, x);int rootY = find(parent, y);if (rootX != rootY) {if (rank[rootX] > rank[rootY]) {parent[rootY] = rootX;} else if (rank[rootX] < rank[rootY]) {parent[rootX] = rootY;} else {parent[rootY] = rootX;rank[rootX]++;}}}// 主函数用于测试 public static void main(String[] args) {KruskalMST graph = new KruskalMST(4);graph.addEdge(0, 1, 10);graph.addEdge(1, 3, 15);graph.addEdge(0, 2, 6);graph.addEdge(2, 3, 4);List<Edge> mst = graph.findMST();for (Edge edge : mst) {System.out.println(edge.src + " - " + edge.dest + ": " + edge.weight);}// 输出: // 2 - 3: 4 // 0 - 2: 6 // 0 - 1: 10 // 最小生成树的构建避免了环的形成。 }
}
4. 动态连通性问题
问题:处理动态连通性查询和合并操作。
实例:在一种动态环境中运行,始终保持对连通性的跟踪。
代码:
public class DynamicConnectivity {private int[] parent;private int[] rank;public DynamicConnectivity(int size) {parent = new int[size];rank = new int[size];for (int i = 0; i < size; i++) {parent[i] = i; // 初始化每个节点的父节点为自己 rank[i] = 0; // 初始化秩为0 }}// 查找并进行路径压缩 public int find(int x) {if (parent[x] != x) {parent[x] = find(parent[x]);}return parent[x];}// 联合操作 public void union(int x, int y) {int rootX = find(x);int rootY = find(y);if (rootX != rootY) {if (rank[rootX] > rank[rootY]) {parent[rootY] = rootX;} else if (rank[rootX] < rank[rootY]) {parent[rootX] = rootY;} else {parent[rootY] = rootX;rank[rootX]++;}}}// 检查两个节点是否在同一连通分量内 public boolean isConnected(int x, int y) {return find(x) == find(y);}// 主函数用于测试 public static void main(String[] args) {DynamicConnectivity dc = new DynamicConnectivity(5);dc.union(0, 1);dc.union(1, 2);System.out.println(dc.isConnected(0, 2)); // 输出 true System.out.println(dc.isConnected(0, 3)); // 输出 false dc.union(2, 3);System.out.println(dc.isConnected(0, 3)); // 输出 true }
}
这些实例展示了如何应用并查集解决一些常见的动态连通性问题,以及如何通过高效的合并和查找来提高性能。
5.思考应用
有些问题像动态连通性、社交网络中的朋友圈判断等都可以使用并查集来高效解决。特别是在需要动态地合并集合并频繁地查询彼此是否连通时,并查集是理想的选择。例如:
- 社交网络:确定任何两个人是否属于同一个社交圈。
- 电网连通性:判断两座城市是否通过电网连接。
通过这些例子,可以看到并查集以其高效的合并和查询能力,从简单集合操作到复杂图结构都有十分广泛的应用。
相关文章:
经典算法思想--并查集
前言 (最近在学习Java,所有函数都是用Java语言来书写的)前言部分是一些前提储备知识 在并查集(Union-Find)数据结构中,rank(中文称为“秩”)是用来表示树的高度或深度的一种辅助信息…...
挑战Java面试题复习第2天,百折不挠
挑战第 2 天 ArrayList和linkedList的区别HashMap和HashTable的区别Collection 与 Collections 的区别Java的四种引用泛型常用特点 ArrayList和linkedList的区别 底层数据结构: ArrayList:基于动态数组实现,支持快速随机访问。LinkedList&a…...
【vue之道】
vue之道 1. 一生二,二生万物思想2. 变化之律3. 变化之实在哪?4.而后学于形乃已!4.1 展示之形变4.2 动之气谓之指令4.3 血之养分的载体,于vue之绑定载具4.4 vue之道(万法规一篇) 1. 一生二,二生万…...
基于麻雀优化算法SSA的CEEMDAN-BiLSTM-Attention的预测模型
往期精彩内容: 时序预测:LSTM、ARIMA、Holt-Winters、SARIMA模型的分析与比较 全是干货 | 数据集、学习资料、建模资源分享! EMD、EEMD、FEEMD、CEEMD、CEEMDAN的区别、原理和Python实现(一)EMD-CSDN博客 EMD、EEM…...
Linux:指令再认识
文章目录 前言一、知识点1. Linux下一切皆文件,也就是说显示器也是一种文件2. 指令是什么?3. ll 与 ls -l4. 日志5. 管道6. 时间戳 二、基本指令1. man指令2. cp指令3. mv指令4. 查看文件1)cat/tac指令——看小文件2)more/less指令…...
PHP如何抛出和接收错误
在PHP中,抛出和接收错误通常涉及异常处理机制,以及错误和异常的处理函数。以下是如何在PHP中抛出和接收错误的详细指南: 抛出错误(异常) 在PHP中,你可以使用throw关键字来抛出一个异常。这通常在你检测到…...
计算机网络:网络层 —— IPv4 地址的应用规划
文章目录 IPv4地址的应用规划定长的子网掩码变长的子网掩码 IPv4地址的应用规划 IPv4地址的应用规划是指将给定的 IPv4地址块 (或分类网络)划分成若干个更小的地址块(或子网),并将这些地址块(或子网)分配给互联网中的不同网络,进而可以给各网络中的主机…...
Mongodb命令大全
Mongodb命令大全 一、数据库相关命令二、集合相关命令三、文档(数据)相关命令1、_id 字段说明2、查询2.1、 查询操作符2.2、内嵌文档查询2.3、数组文档查询2.4、去重查询2.5、查询排序 sort2.6、分页查询2.7、指定列投影查询返回2.8、查询统计个数 count 3、聚合查询3.1、查询用…...
宇视设备视频平台EasyCVR视频融合平台果园/鱼塘/养殖场/菜园有电没网视频监控方案
在那些有电无网的偏远地区,如果园、鱼塘、养殖场或菜园,视频监控的实现面临着独特的挑战。宇视设备视频平台EasyCVR提供了一种创新的解决方案,通过结合太阳能供电和4G摄像头技术,有效地解决了这些场景下的监控需求。 在有电没网的…...
面试题:ABCD四个线程,A线程最后执行
我觉得是一个很高频的面试题,ABCD四个线程,A线程要等到BCD线程执行完再执行,怎么做 因为我刚复习完AQS,所以立马想到了CountDownLatch,但是看面试官反应他最想听到的应该是join方法,所以面试后就总结了几种…...
代码随想录算法训练营第46期Day43
leetcode.322零钱兑换 class Solution { public: //无限个硬币->完全背包int coinChange(vector<int>& coins, int amount) {vector<int> dp(10010,INT_MAX);//dp代表的在某个数值下最小的硬币数,要求是最小的硬币数,所以初始值要尽可…...
前端处理API接口故障:多接口自动切换的实现方案
因为在开发APP,一个接口如果不通(被挂了)又不能改了重新打包让用户再下载软件更新,所以避免这种情况,跟后端讨论多备用接口地址自动切换的方案,自动切换到备用的接口地址,并保证后续所有的请求都…...
多租户架构的全景分析(是什么?基本概念、实现策略、资源管理和隔离、数据安全与隔离、性能优化、扩展性与升级、案例研究)
文章目录 1. 多租户的基本概念2. 多租户的实现策略2.1 独立数据库模式2.2 共享数据库-独立Schema模式2.3 共享数据库-共享Schema模式 3. 资源管理和隔离4. 数据安全与隔离5. 性能优化6. 扩展性与升级7. 案例研究总结 多租户架构在云计算和SaaS应用中越来越流行,因为…...
Git使用问题汇总附带解决方法(持续更新)
Git使用问题汇总附带解决方法 一 git pull 代码时报错: Auto packing the repository in background for optimum performance. See “git help gc“ 一 git pull 代码时报错: Auto packing the repository in background for optimum performance. See …...
Spring Boot驱动的植物健康监测革命
1系统概述 1.1 研究背景 随着计算机技术的发展以及计算机网络的逐渐普及,互联网成为人们查找信息的重要场所,二十一世纪是信息的时代,所以信息的管理显得特别重要。因此,使用计算机来管理植物健康系统的相关信息成为必然。开发合适…...
element 中 el-dialog 在不同的文件中使用
在实际中工作,我们经常需要使用 el-dialog 来做一个弹框的功能。最常见的就是在父组件中点击一个按纽,然后弹出一个框。而这个框就是子组件。同时,父子组件是分布在不同的文件中。 <!--父组件--> <template> <div> <…...
QT中采用QCustomPlot 实现将buffer中的数据绘制成折线图,并且图形随着数据更新而更新
QT中采用QCustomPlot 实现将buffer中的数据绘制成折线图,并且图形随着数据更新而更新 为了在 Qt 中将缓冲区的数据动态绘制成折线图,并随着数据的更新而实时更新,可以使用 QCustomPlot 或 Qt 自带的绘图功能,比如 QGraphicsView,或者在更简单的情况下使用 QPainter 在 QW…...
1688API商品详情接口如何获取
获取 1688API商品详情接口主要有以下步骤: 一、注册开发者账号: 访问 1688 开放平台,进行开发者账号注册。这是获取 API 接口使用权限的第一步,注册信息要确保真实准确。 二、了解接口规范和政策: 在 1688 开放平台…...
pytorch + d2l环境配置
文章目录 前言一、安装软件二、配置具体环境 前言 一直想写一篇 pytorch d2l的深度学习环境配置。但一直都不是很顺利,配置过很多次,都会遇到一些各种依赖项的兼容性问题。但这个是没有办法的,各种开源包都在不断维护过程中,版本…...
Go使用exec.Command() 执行脚本时出现:file or directory not found
使用 Go 提供的 exec.Command() 执行脚本时出现了未找到脚本的 bug,三个排查思路 : exec.Command(execName, args…) 脚本名字不允许相对路径 exec.Command(execName, args…) execName 只能有脚本名,不允许出现参数 如果你是使用 Windows …...
Python|GIF 解析与构建(5):手搓截屏和帧率控制
目录 Python|GIF 解析与构建(5):手搓截屏和帧率控制 一、引言 二、技术实现:手搓截屏模块 2.1 核心原理 2.2 代码解析:ScreenshotData类 2.2.1 截图函数:capture_screen 三、技术实现&…...
Android Wi-Fi 连接失败日志分析
1. Android wifi 关键日志总结 (1) Wi-Fi 断开 (CTRL-EVENT-DISCONNECTED reason3) 日志相关部分: 06-05 10:48:40.987 943 943 I wpa_supplicant: wlan0: CTRL-EVENT-DISCONNECTED bssid44:9b:c1:57:a8:90 reason3 locally_generated1解析: CTR…...
synchronized 学习
学习源: https://www.bilibili.com/video/BV1aJ411V763?spm_id_from333.788.videopod.episodes&vd_source32e1c41a9370911ab06d12fbc36c4ebc 1.应用场景 不超卖,也要考虑性能问题(场景) 2.常见面试问题: sync出…...
智慧工地云平台源码,基于微服务架构+Java+Spring Cloud +UniApp +MySql
智慧工地管理云平台系统,智慧工地全套源码,java版智慧工地源码,支持PC端、大屏端、移动端。 智慧工地聚焦建筑行业的市场需求,提供“平台网络终端”的整体解决方案,提供劳务管理、视频管理、智能监测、绿色施工、安全管…...
无法与IP建立连接,未能下载VSCode服务器
如题,在远程连接服务器的时候突然遇到了这个提示。 查阅了一圈,发现是VSCode版本自动更新惹的祸!!! 在VSCode的帮助->关于这里发现前几天VSCode自动更新了,我的版本号变成了1.100.3 才导致了远程连接出…...
04-初识css
一、css样式引入 1.1.内部样式 <div style"width: 100px;"></div>1.2.外部样式 1.2.1.外部样式1 <style>.aa {width: 100px;} </style> <div class"aa"></div>1.2.2.外部样式2 <!-- rel内表面引入的是style样…...
是否存在路径(FIFOBB算法)
题目描述 一个具有 n 个顶点e条边的无向图,该图顶点的编号依次为0到n-1且不存在顶点与自身相连的边。请使用FIFOBB算法编写程序,确定是否存在从顶点 source到顶点 destination的路径。 输入 第一行两个整数,分别表示n 和 e 的值(1…...
OPenCV CUDA模块图像处理-----对图像执行 均值漂移滤波(Mean Shift Filtering)函数meanShiftFiltering()
操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C11 算法描述 在 GPU 上对图像执行 均值漂移滤波(Mean Shift Filtering),用于图像分割或平滑处理。 该函数将输入图像中的…...
使用 Streamlit 构建支持主流大模型与 Ollama 的轻量级统一平台
🎯 使用 Streamlit 构建支持主流大模型与 Ollama 的轻量级统一平台 📌 项目背景 随着大语言模型(LLM)的广泛应用,开发者常面临多个挑战: 各大模型(OpenAI、Claude、Gemini、Ollama)接口风格不统一;缺乏一个统一平台进行模型调用与测试;本地模型 Ollama 的集成与前…...
鸿蒙DevEco Studio HarmonyOS 5跑酷小游戏实现指南
1. 项目概述 本跑酷小游戏基于鸿蒙HarmonyOS 5开发,使用DevEco Studio作为开发工具,采用Java语言实现,包含角色控制、障碍物生成和分数计算系统。 2. 项目结构 /src/main/java/com/example/runner/├── MainAbilitySlice.java // 主界…...
