只是不愿随波逐流 ...|

lidongdongdong~

园龄:2年7个月粉丝:14关注:8

15、并查集

内容来自刘宇波老师算法与数据结构体系课

1、QuickFind

public interface UF {
int getSize();
/**
* 查看元素 p 和元素 q 是否所属一个集合
*/
boolean isConnected(int p, int q);
/**
* 合并元素 p 和元素 q 所属的集合
*/
void unionElements(int p, int q);
}
/**
* QuickFind
* isConnected(p, q): O(1)
* unionElements(p, q): O(n)
*/
public class UnionFind1 implements UF {
private final int[] id;
public UnionFind1(int size) {
id = new int[size];
for (int i = 0; i < size; i++) id[i] = i;
}
@Override
public int getSize() {
return id.length;
}
/**
* 查找元素 p 所对应的集合编号
*/
private int find(int p) {
if (p < 0 || p >= id.length) throw new IllegalArgumentException("p is out of bound.");
return id[p];
}
@Override
public boolean isConnected(int p, int q) {
return find(p) == find(q);
}
@Override
public void unionElements(int p, int q) {
int pID = find(p);
int qID = find(q);
if (pID == qID) return;
for (int i = 0; i < id.length; i++) {
if (id[i] == pID) id[i] = qID;
}
}
}

2、QuickUnion

/**
* QuickUnion
* isConnected(p, q): O(h)
* unionElements(p, q): O(h)
* h 为树的高度, 极端情况下, 并查集中的树可能会退化成一个链表
*/
public class UnionFind2 implements UF {
private final int[] parent;
public UnionFind2(int size) {
parent = new int[size];
for (int i = 0; i < size; i++) parent[i] = i;
}
@Override
public int getSize() {
return parent.length;
}
/**
* 查找元素 p 所对应的集合编号
*/
private int find(int p) {
if (p < 0 || p >= parent.length) throw new IllegalArgumentException("p is out of bound.");
while (p != parent[p]) p = parent[p];
return p;
}
@Override
public boolean isConnected(int p, int q) {
return find(p) == find(q);
}
@Override
public void unionElements(int p, int q) {
int pRoot = find(p);
int qRoot = find(q);
if (pRoot == qRoot) return;
parent[pRoot] = qRoot;
}
}

3、基于 size 进行优化

/**
* 基于 size 的优化
* sz[i] 代表以 i 为根的集合中的元素个数
* 让 "元素个数少的集合" 合并到 "元素个数多的集合" 上, 使得合并后的新树, 深度尽量不要增加
*/
public class UnionFind3 implements UF {
private final int[] parent;
private final int[] sz; // sz[i] 代表以 i 为根的集合中的元素个数
public UnionFind3(int size) {
parent = new int[size];
sz = new int[size];
for (int i = 0; i < size; i++) {
parent[i] = i;
sz[i] = 1;
}
}
@Override
public int getSize() {
return parent.length;
}
/**
* 查找元素 p 所对应的集合编号
*/
private int find(int p) {
if (p < 0 || p >= parent.length) throw new IllegalArgumentException("p is out of bound.");
while (p != parent[p]) p = parent[p];
return p;
}
@Override
public boolean isConnected(int p, int q) {
return find(p) == find(q);
}
@Override
public void unionElements(int p, int q) {
int pRoot = find(p);
int qRoot = find(q);
if (pRoot == qRoot) return;
// 让 "元素个数少的集合" 合并到 "元素个数多的集合" 上: 合并后的新树, 深度尽量不要增加
if (sz[pRoot] < sz[qRoot]) {
parent[pRoot] = qRoot;
sz[qRoot] += sz[pRoot];
} else {
parent[qRoot] = pRoot;
sz[pRoot] += sz[qRoot];
}
}
}

4、基于 rank 进行优化

/**
* 基于 rank 的优化
* rank[i] 代表以 i 为根的集合所表示的树的层数
* 让 "层数低的集合" 合并到 "层数高的集合" 上, 使得合并后的新树, 深度尽量不要增加
*/
public class UnionFind4 implements UF {
private final int[] parent;
private final int[] rank; // rank[i] 代表以 i 为根的集合所表示的树的层数
public UnionFind4(int size) {
parent = new int[size];
rank = new int[size];
for (int i = 0; i < size; i++) {
parent[i] = i;
rank[i] = 1;
}
}
@Override
public int getSize() {
return parent.length;
}
/**
* 查找元素 p 所对应的集合编号
*/
private int find(int p) {
if (p < 0 || p >= parent.length) throw new IllegalArgumentException("p is out of bound.");
while (p != parent[p]) p = parent[p];
return p;
}
@Override
public boolean isConnected(int p, int q) {
return find(p) == find(q);
}
@Override
public void unionElements(int p, int q) {
int pRoot = find(p);
int qRoot = find(q);
if (pRoot == qRoot) return;
// 让 "层数低的集合" 合并到 "层数高的集合" 上: 合并后的新树, 深度尽量不要增加
if (rank[pRoot] < rank[qRoot]) parent[pRoot] = qRoot;
else if (rank[qRoot] < rank[pRoot]) parent[qRoot] = pRoot;
else {
parent[pRoot] = qRoot;
rank[qRoot] += 1;
}
}
}

5、路径压缩 1

image

/**
* 基于 rank 的优化 + 路径压缩 1
* rank[i] 代表以 i 为根的集合所表示的树的层数的排名
* 让 "层数排名低的集合" 合并到 "层数排名高的集合" 上, 使得合并后的新树, 深度尽量不要增加
* 平均来看, 近乎是 O(1) 级别的
*/
public class UnionFind5 implements UF {
private final int[] parent;
private final int[] rank; // rank[i] 代表以 i 为根的集合所表示的树的层数的排名
public UnionFind5(int size) {
parent = new int[size];
rank = new int[size];
for (int i = 0; i < size; i++) {
parent[i] = i;
rank[i] = 1;
}
}
@Override
public int getSize() {
return parent.length;
}
/**
* 查找元素 p 所对应的集合编号, 同时压缩树的高度使其尽可能的矮(路径压缩)
*/
private int find(int p) {
if (p < 0 || p >= parent.length) throw new IllegalArgumentException("p is out of bound.");
while (p != parent[p]) {
parent[p] = parent[parent[p]]; // 路径压缩, p 指向 p 的爷爷, 不用维护 rank
p = parent[p];
}
return p;
}
@Override
public boolean isConnected(int p, int q) {
return find(p) == find(q);
}
@Override
public void unionElements(int p, int q) {
int pRoot = find(p);
int qRoot = find(q);
if (pRoot == qRoot) return;
// 让 "层数排名低的集合" 合并到 "层数排名高的集合" 上: 合并后的新树, 深度尽量不要增加
if (rank[pRoot] < rank[qRoot]) parent[pRoot] = qRoot;
else if (rank[qRoot] < rank[pRoot]) parent[qRoot] = pRoot;
else {
parent[pRoot] = qRoot;
rank[qRoot] += 1;
}
}
}

5、路径压缩 2

image

/**
* 基于 rank 的优化 + 路径压缩 2
* rank[i] 代表以 i 为根的集合所表示的树的层数的排名
* 让 "层数排名低的集合" 合并到 "层数排名高的集合" 上, 使得合并后的新树, 深度尽量不要增加
* 平均来看, 近乎是 O(1) 级别的
*/
public class UnionFind6 implements UF {
private final int[] parent;
private final int[] rank; // rank[i] 代表以 i 为根的集合所表示的树的层数的排名
public UnionFind6(int size) {
parent = new int[size];
rank = new int[size];
for (int i = 0; i < size; i++) {
parent[i] = i;
rank[i] = 1;
}
}
@Override
public int getSize() {
return parent.length;
}
/**
* 查找元素 p 所对应的集合编号, 同时压缩树的高度使其尽可能的矮(路径压缩)
*/
private int find(int p) {
if (p < 0 || p >= parent.length) throw new IllegalArgumentException("p is out of bound.");
if (p == parent[p]) return p;
parent[p] = find(parent[p]);
return parent[p];
}
@Override
public boolean isConnected(int p, int q) {
return find(p) == find(q);
}
@Override
public void unionElements(int p, int q) {
int pRoot = find(p);
int qRoot = find(q);
if (pRoot == qRoot) return;
// 让 "层数排名低的集合" 合并到 "层数排名高的集合" 上: 合并后的新树, 深度尽量不要增加
if (rank[pRoot] < rank[qRoot]) parent[pRoot] = qRoot;
else if (rank[qRoot] < rank[pRoot]) parent[qRoot] = pRoot;
else {
parent[pRoot] = qRoot;
rank[qRoot] += 1;
}
}
}

6、总结

并查集用于解决连接问题
1、QuickFind、QuickUnion
2、基于 size 的优化:让 "元素个数少的集合" 合并到 "元素个数多的集合" 上,使得合并后的新树,深度尽量不要增加
3、基于 rank 的优化:让 "层数低的集合" 合并到 "层数高的集合" 上,使得合并后的新树,深度尽量不要增加
4、路径压缩:压缩树的高度使其尽可能的矮
posted @   lidongdongdong~  阅读(17)  评论(0编辑  收藏  举报
点击右上角即可分享
微信分享提示
评论
收藏
关注
推荐
深色
回顶
展开