# 实现
当结点是连续的时候,使用数组实现
class UnionFind{
int[] parents;
UnionFind(int size){
parents = new int[size];
// 初始化父节点,所有结点指向自己
for(int i = 0; i < size; i++){
parents[i] = i;
}
}
// 带路径压缩的查找,把所有子结点都指向其最终父结点
public int find(int x){
if(parents[x] == x)
return x;
return parents[x] = find(parents[x]);
}
// 将结点x与结点y所在的连通域合并,其实就是把y的父结点指向x的父结点
public void union(int x, int y){
int xParent = find(x);
int yParent = find(y);
if(xParent != yParent){
parents[yParent] = xParent;
}
}
// 判断结点x和y是否在一个连通域
public boolean isConnect(int x, int y){
return find(x) == find(y);
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
当结点是离散的时候,用哈希表实现,注意find方法与数组实现的不同。
class UnionFind{
Map<Integer, Integer> parents;
UnionFind(int[] nums){
parents = new HashMap<>();
for(int num : nums){
parents.put(num, num);
}
}
public int find(int x){
if(x == parents.get(x)){
return x;
}
int parent = find(parents.get(x));
parents.put(x, parent);
// return parents.put(x, find(parents.get(x))); 这种写法是错误的,put返回的是map中x位置被替换的旧值
return parent;
}
public void union(int x, int y){
int xParent = find(x);
int yParent = find(y);
if(xParent != yParent){
parents.put(yParent, xParent);
}
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28