【java算法】并查集

算法适用

给定一些操作,查找联通块/查找集合数/查找环的个数

节点初始化

创建一维数组,大小为n+1,数组索引代表节点名,数组值代表指向节点

查找父节点

若当前节点指向自身,为根节点,找到并返回
否则,顺着指向的节点(数组索引位置的值)继续寻找

链接节点,合并集合

int a_value=find(a);
int b_value=find(b);
num[a_value]=b_value;
若根节点相同,可以不连接

同时,若根节点相同,说明图中出现了环

路径压缩

本质是存储搜索的结果

public int find(int a) {
	if (num[a]==a) return a;
	else {
	num[a]=find(num[a]);//路径压缩
		return num[a];
	}
}

返回集合个数

指向自身的节点为一个集合的父节点,统计数组值等于自身的节点个数

应用实例

经典并查集题目合根植物
一道搜索题,也可以用并查集来解细胞数量
使用面向对象的形式封装好

import java.util.*;
public class 手搓并查集 {
	public static class bcj{
		static int num[];
		public bcj(int n) {//参数是并查集的大小
			// TODO 自动生成的构造函数存根
		 num=new int[n+1];
		 for (int i = 1; i <=n; i++) {
			num[i]=i;//初始化 所有的节点祖先指向自身
		}
		}
		
	public int find(int a) {
		if (num[a]==a) return a;
		else {
			num[a]=find(num[a]);//路径压缩,find函数的参数一定要填数组的值,一层一层找上去,否则无限循环
			return num[a];
		}
	}
	public void add(int a,int b) {//链接节点
		int a_value=find(a);
		int b_value=find(b);
		num[a_value]=b_value;//次序不重要
	}
	public int sumnode() {
		int ans=0;
		for (int i = 1; i < num.length; i++) {
			if(num[i]==i) ans++;
		}
		return ans;
	}
	}
public static void main(String[] args) {
	Scanner scan=new Scanner(System.in);
	bcj a=new bcj(scan.nextInt()*scan.nextInt());
	int N=scan.nextInt();
	for (int i = 0; i < N; i++) {
		a.add(scan.nextInt(), scan.nextInt());
	}
	System.out.println(a.sumnode());
	scan.close();
}
}

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值