数据结构—并查集《下》
上一节主要介绍了 Quick Find 的思想和代码实现,本节要介绍的是 Quick Union的实现和代码实现。
Quick Union - Union
-
Quick Union的union(v1, v2):让 v1 的根节点指向 v2 的根节点
public void union(int v1, int v2) { int p1 = find(v1); int p2 = find(v2); if (p1 == p2) return; parents[p1] = p2; }
-
时间复杂度:O(logn)
public int find(int v) { rangeCheck(v); while (v != parents[v]) { v = parents[v]; } return v; } find(0) == 2 find(1) == 2 find(2) == 2 find(3) == 3 find(4) == 2
Quick Union - 优化
-
在 Union 的过程中,可能会出现树不平衡的情况,甚至退化成链表
-
有2种常见的优化方案
-
基于size的优化:元素少的树嫁接到元素多的树
-
基于rank的优化:矮的树嫁接到高的树
Quick Union - 1.基于 size 的优化
sizes = new int[capacity]; for (int i = 0; i < sizes.length; i++) { size[i] = 1; } private int[] sizes; public void union(int v1, int v2) { int p1 = find(v1); int p2 = find(v2); if (p1 == p2) return; if (sizes[p1] < sizes[p2]) { parents[p1] = p2; sizes[p2] += sizes[p1]; } else { parents[p2] = p1; sizes[p1] += sizs[p2]; } }
基于 size 的优化,也可能会存在树不平衡的问题
Quick Union - 2.基于 rank 的优化
ranks = new int[capacity]; for (int i = 0; i < ranks.length; i++) { ranks[i] = 1; } private int[] ranks; public void union(int v1, int v2) { int p1 = find(v1); int p2 = find(v2); if (p1 == p2) return; if (ranks[p1] < ranks[p2]) { parents[p1] = p2; } else if (ranks[p2] < ranks[p1]) { parents[p2] = p1; } else { parents[p1] = p2; ranks[p2]++; } }
压缩路径(Path Compression)
-
虽然有了基于 rank 的优化,树会相对平衡一点
-
但是随着 Union 次数的增多,树的高度依然会越来越高,导致find操作变慢,尤其是底层节点(因为 find 是不断向上找到根节点)
-
什么是路径压缩?
-
在 find 时使路径上的所有节点都指向根节点,从而降低树的高度
public int find(int v) { rangeCheck(v); if (parents[v] != v) { parents[v] = find(parents[v]); } return parents[v]; }
-
路径压缩使路径上的所有节点都指向根节点,所以实现成本稍高
-
还有2种更有效的做法,但不能降低树高,实现成本也比路径压缩低
-
路径分裂(Path Spliting)
-
路径减半(Path Halving)
-
路径分裂、路径减半的效率差不多,但是都比路径压缩要好
1.路径分裂(Path Spliting)
-
路径分裂:使路径上的每个节点都指向其祖父节点(parent的parent)
public int find(int v) { rangeCheck(v); while (v != parents[v]) { int parent = parents[v]; parents[v] = parents[parent]; v = parent; } return v; }
2.路径减半(Path Halving)
-
路径减半:是路径上每隔一个节点就指向其祖父节点(parent的parent)
public int find(int v) { rangeCheck(v); while (v != parents[v]) { parents[v] = parents[parents[v]]; v = parents[v]; } return v; }
总结
今天主要讲解了 Quick Union 的原理和代码实现。然后讲解了基于 size 和基于 rank 的优化。最后简单的介绍了下路径压缩。并查集到这就讲完了,希望给大家的知识库增加一些新的知识储备。
欢迎关注【无量测试之道】公众号,回复【领取资源】
Python编程学习资源干货、
Python+Appium框架APP的UI自动化、
Python+Selenium框架Web的UI自动化、
Python+Unittest框架API自动化、
资源和代码 免费送啦~
文章下方有公众号二维码,可直接微信扫一扫关注即可。
备注:我的个人公众号已正式开通,致力于测试技术的分享,包含:大数据测试、功能测试,测试开发,API接口自动化、测试运维、UI自动化测试等,微信搜索公众号:“无量测试之道”,或扫描下方二维码:
添加关注,让我们一起共同成长!