Java数据结构之并查集的实现
作者:chaojilaji 发布时间:2023-06-29 06:58:32
并查集就是将原本不在一个集合里面的内容合并到一个集合中。
在实际的场景中用处不多。
除了出现在你需要同时去几个集合里面查询,避免出现查询很多次,从而放在一起查询的情况。
下面简单实现一个例子,我们来举例说明一下什么是并查集,以及究竟并查集解决了什么问题。
代码解析
package com.chaojilaji.book.andcheck;
public class AndCheckSet {
public static Integer getFather(int[] father, int u) {
if (father[u] != u) {
father[u] = getFather(father, father[u]);
}
return father[u];
}
public static void join(int[] father,int x, int y) {
int fx = getFather(father,x);
int fy = getFather(father,y);
if (fx != fy){
father[fx] = fy;
}
}
public static void main(String[] args) {
int n = 10;
int[] a = new int[n];
for (int i = 0;i<n;i++){
a[i] = i; // 初始化定义一百个集合
}
for (int i=0;i<n;i++){
System.out.println(i+" "+getFather(a, i)); // 对于每个集合,父节点都是自己
}
}
}
首先,我们定义了两个函数,分别为getFather和join,分别表示获取u所在集合的根以及合并两个集合。
先来看看getFather方法
public static Integer getFather(int[] father, int u) {
if (father[u] != u) {
father[u] = getFather(father, father[u]);
}
return father[u];
}
是找出值u所在集合的根节点是多少。一般来说,father[u]如果等于u本身,那么说明u所在节点就是根节点,而这个算法是直到相等才退出,也就是说,对于从u到根节点的每个节点的father都被直接置为根节点,同时返回了当前节点的根节点。
然后来看看join方法
public static void join(int[] father,int x, int y) {
int fx = getFather(father,x);
int fy = getFather(father,y);
if (fx != fy){
father[fx] = fy;
}
}
分别找出x和y两个节点所在集合的根节点,如果根节点不一样,则将其中一个节点的father节点置为另一个节点即可,这样就合并成了一个集合。
代码应用
public static void main(String[] args) {
int n = 10;
int[] a = new int[n];
for (int i = 0;i<n;i++){
a[i] = i; // 初始化定义n个集合
}
for (int i=0;i<n;i++){
System.out.println(i+" "+getFather(a, i)); // 对于每个集合,父节点都是自己
}
System.out.println("-------------------------");
join(a,0,1); // 合并 0 和 1
for (int i=0;i<n;i++){
System.out.println(i+" "+getFather(a, i));
}
// 由于合并了0和1,所以集合变成了9个,节点0和节点1的根都是节点1
System.out.println("-------------------------");
join(a,2,3); // 合并 2 和 3
for (int i=0;i<n;i++){
System.out.println(i+" "+getFather(a, i));
}
// 由于合并了2和3,所以集合变成8个,节点2和节点3的根都是节点3
System.out.println("-------------------------");
join(a,1,3); // 合并 1 和 3
for (int i=0;i<n;i++){
System.out.println(i+" "+getFather(a, i));
}
// 由于合并了1和3,所以集合变成7个,节点0,1,2,3的根都是节点3
}
首先,我们定义了n个集合,这n个集合的值是0~n-1,然后此时他们的父节点均等于他们本身,所以这就是n个独立的集合,结果如下
0的父节点为 0 1的父节点为 1 2的父节点为 2 3的父节点为 3 4的父节点为 4 5的父节点为 5 6的父节点为 6 7的父节点为 7 8的父节点为 8 9的父节点为 9
然后调用 join(a,0,1)合并0集合和1集合,再输出节点父集合情况为
0的父节点为 1 1的父节点为 1 2的父节点为 2 3的父节点为 3 4的父节点为 4 5的父节点为 5 6的父节点为 6 7的父节点为 7 8的父节点为 8 9的父节点为 9
可以看见,由于合并了0和1,所以集合变成了9个,节点0和节点1的根都是节点1。
然后调用 join(a,2,3) 合并2集合和3集合,输出节点父集合情况为
0的父节点为 1 1的父节点为 1 2的父节点为 3 3的父节点为 3 4的父节点为 4 5的父节点为 5 6的父节点为 6 7的父节点为 7 8的父节点为 8 9的父节点为 9
可以看见,由于合并了2和3,所以集合变成8个,节点2和节点3的根都是节点3。
最后,我们再调用join(a,1,3) 合并1集合和3集合,输出节点父集合情况为
0的父节点为 3 1的父节点为 3 2的父节点为 3 3的父节点为 3 4的父节点为 4 5的父节点为 5 6的父节点为 6 7的父节点为 7 8的父节点为 8 9的父节点为 9
可以看出来,由于合并了1和3,所以集合变成7个,节点0,1,2,3的根都是节点3。
实际应用
代码的层面来讲,并查集很好实现。但是我们却也可以发现,其应用场景似乎非常局限。
首先,我们需要定义出一个father[x] = x的数组,然后我们再去合并。似乎很难想到应用场景。
那么我们可以想象一个场景,现在有个网络拓扑图,里面有n和网络设施设备,然后又给了你这n个设施设备之间的连接关系,问你一共有多少个局部联通网。
对于这个问题,我们就可以首先定义每个设备自己跟自己相连,然后每出现一条边,就对这两个设备采取join操作。最终我们在遍历完所有的节点,得到多少个不同的father,即表示有多少个不同的局部联通网。
这样的问题还可以延伸到我们的人际交友圈,首先每个人都是单独的集合,在不断认识人的过程中,产生连通性。最终让你确认一共有多少个不互通的人际圈。
所以你会发现,这本质上就是求图论中连通块的个数。
来源:https://juejin.cn/post/7055647928911659044


猜你喜欢
- 本文实例讲述了WinForm实现状态栏跑马灯效果的方法。分享给大家供大家参考,具体如下:using System;using System.
- 本文实例讲述了C#数据结构之单链表(LinkList)实现方法。分享给大家供大家参考,具体如下:这里我们来看下“单链表(LinkList)”
- C#WinForm程序设计之图片浏览器,这次我们一起做一个图片查看器,这个图片查看器的原始图如下:我们首先来介绍一下这个原始图的构成:左边上
- java里有数字long来表示大的整数,如果两个数字的范围超过了long,要做加法算法怎么做呢?这个问题在面试中经常碰到,如果之前没有经历的
- 创建一个脚本,输入下面的代码。脚本不需要添加到任何物体上,globle变量可以跨场景全局调用。using System.Collection
- 一、下载https://www.eclipse.org/downloads/download.php?file=/oomph/epp/202
- 在APP项目的开发过程中,经常会用到分享图片的功能,有时候还需要根据当前用户信息获取指定的分享图片,比如要求在用户分享图中显示用户名、Uid
- 本文实例为大家分享了Java使用组件编写窗口下载网上文件的具体代码,供大家参考,具体内容如下如图实现代码:package com.rain.
- 最近碰到这么个恶心的问题问题:有个arr文件被放到Module A中引用,现在Module B又依赖了Module A,则在编译过程中会发生
- 系列文章已完成,目录如下:jdk-logging log4j logback日志系统实现机制原理介绍commons-lo
- 在使用SpringSecurity中,大伙都知道默认的登录数据是通过key/value的形式来传递的,默认情况下不支持JSON格式的登录数据
- 好问题。答案就是这篇文章的题目所建议的,这是一种合理的设计。在这种情况下,newInstance()方法是一种“静态工厂方法",让
- 前言MVC模式是目前主流项目的标准开发模式,这种模式下框架的分层结构清晰,主要分为Controller,Service,Dao。分层的结构下
- 今天使用jenkins构建时,报以下错误[ERROR] Failed to execute goal on project saas20:
- Mybatis typeAlias标签在实际的工程之中,类的全限定名称很长,当我们需要大量使用的时候,这非常不方便的,然而mybatis提供
- 前言之所以会有这篇文章,是因为公司的开发环境比较老,寻找一些jar包的时候总是会纠结对应的编译版本,感觉很麻烦,所以写了一个工具类用于读取c
- 一、基本介绍 1、介绍学习很多算法知识,力争做到最优解的学习过程中,很多时候都会遇到PriorityQueue(优先队列)。一个基
- 基本概念 * (Listener): * 用于监听web应用中某些对象、信息的创建、销毁、增加,修改,删除等动作的发生,然后作出相应的响应处
- Spring Boot 内置Tomcat默认端口号为8080,在开发多个应用调试时很不方便,本文介绍了修改 Spring Boot
- 本文实例讲述了Android在JNI中使用ByteBuffer的方法。分享给大家供大家参考。具体如下:一、ByteBuffer 定义在NIO