问题→

动态连通性:当程序从输入中读取了整数对p q时,如果已知的所有整数对都不能说明p和q是相连的,那么则将这一对整数写入到输出中。如果已知的数据可以说明p和q

是相连的,那么程序应该忽略p q这对整数并继续处理输入中的下一对整数。

该问题的应用→

网络,变量名等价性,数字集合等。

设计API→

public class UF
  UF(int N) 以整数标识(0到N-1)初始化N个触点
void union(int p, int q) 在p和q之间添加一条连接
int find(int p) p(0到N-1)所在的分量的标识符
boolean connected(int p, int q) 如果p和q存在于同一个分量中则返回true
int count() 连通分量的数量

为解决动态连通性问题设计算法的任务转化为了实现这份API。

quick-find算法→

public class QuickFindUF {

    private int[] id;     // 分量id(以触点作为索引)
private int count; // 分量数量 public QuickFindUF(int N) { // 初始化分量id数组
count = N;
id = new int[N];
for (int i = 0; i < N; i++) {
id[i] = i;
}
} public int count() {
return count;
} public boolean connected(int p, int q) {
return find(p) == find(q);
} public int find(int p) {
return id[p];
} public void union(int p, int q) { // 将p和q归并到相同的分量中
int pID = find(p);
int qID = find(q); // 如果p和q已经在相同的分量之中则不需要采取任何行动
if (pID == qID) return; // 将p的分量重命名为q的名称
for (int i = 0; i < id.length; i++) {
if (id[i] == pID) id[i] = qID;
} count--;
} public static void main(String[] args) { // 解决由StdIn得到的动态连通性问题
int N = StdIn.readInt(); // 读取触点数量
UF uf = new UF(N); // 初始化N个分量
while (!StdIn.isEmpty()) {
int p = StdIn.readInt();
int q = StdIn.readInt(); // 读取整数对
if (uf.connected(p, q)) continue; // 如果已经连通则忽略
uf.union(p, q); // 归并分量
StdOut.println(p + " " + q); // 打印链接
}
StdOut.println(uf.count() + " components");
} }

QuickFindUF

分析:

· 每次find()调用只需要访问数组一次,而归并两个分量的union()操作访问数组的次数在(N + 3)到(2N + 1)之间。

· 假设最后只得到了一个连通分量,那么这至少需要调用N - 1次union(),即至少(N + 3)(N - 1) ~ N ^ 2次数组访问。

· quick-find算法是平方级别的。

quick-union算法→

public class QuickUnionUF {

    private int[] id;     // 分量id(以触点作为索引)
private int count; // 分量数量 public QuickUnionUF(int N) { // 初始化分量id数组
count = N;
id = new int[N];
for (int i = 0; i < N; i++) {
id[i] = i;
}
} public int count() {
return count;
} public boolean connected(int p, int q) {
return find(p) == find(q);
} public int find(int p) {
// return id[p];
while (p != id[p]) p = id[p];
return p;
} public void union(int p, int q) {
int pRoot = find(p);
int qRoot = find(q);
if (pRoot == qRoot) return; id[pRoot] = qRoot; count--;
} public static void main(String[] args) { // 解决由StdIn得到的动态连通性问题
int N = StdIn.readInt(); // 读取触点数量
UF uf = new UF(N); // 初始化N个分量
while (!StdIn.isEmpty()) {
int p = StdIn.readInt();
int q = StdIn.readInt(); // 读取整数对
if (uf.connected(p, q)) continue; // 如果已经连通则忽略
uf.union(p, q); // 归并分量
StdOut.println(p + " " + q); // 打印链接
}
StdOut.println(uf.count() + " components");
} }

QuickUnionUF

分析:

· 最佳情况的输入,用例的运行时间是线性级别的。

· 最坏情况的输入,用例的运行时间是平方级别的。

加权quick-union算法→

public class WeightedQuickUnionUF {
private int[] id;
private int[] sz;
private int count; public WeightedQuickUnionUF(int N) {
count = N;
id = new int[N];
for (int i = 0; i < N; i++) id[i] = i;
sz = new int[N];
for (int i = 0; i < N; i++) sz[i] = 1;
} public int count() {
return count;
} public boolean connected(int p, int q) {
return find(p) == find(q);
} public int find(int p) { // 跟随链接找到根节点
while (p != id[p]) p = id[p];
return p;
} public void union(int p, int q) {
int i = find(p);
int j = find(q);
if (i == j) return;
// 将小树的根节点连接到大树的根节点
if (sz[i] < sz[j]) {
id[i] = j;
sz[j] += sz[i];
} else {
id[j] = i;
sz[i] += sz[j];
}
count--;
} public static void main(String[] args) { // 解决由StdIn得到的动态连通性问题
int N = StdIn.readInt(); // 读取触点数量
UF uf = new UF(N); // 初始化N个分量
while (!StdIn.isEmpty()) {
int p = StdIn.readInt();
int q = StdIn.readInt(); // 读取整数对
if (uf.connected(p, q)) continue; // 如果已经连通则忽略
uf.union(p, q); // 归并分量
StdOut.println(p + " " + q); // 打印链接
}
StdOut.println(uf.count() + " components");
} }

WeightedQuickUnionUF

分析:

· 记录每一棵树的大小并总是将较小的树连接到较大的树上。

· 加权quick-union算法是对数级别的。

最优算法→

路径压缩的加权quick-union算法是最优的算法,但并非所有的操作都能在常数时间内完成。

均摊成本的图像→

· 对于quick-find算法:累计平均值一开始较高,后来开始下降,但仍保持了相对较高的水平。

· 对于quick-union算法:累计平均值在初始阶段较低,后期增长很明显。

· 对于加权quick-union算法:没有任何昂贵的操作,均摊成本也很低。

展望→

讨论问题时的基本步骤:

· 完整而详细地定义问题,找出解决问题所必需的基本抽象操作并定义一份API。

· 简洁地实现一种初级算法,给出一个精心组织的开发用例并使用实际数据作为输入。

· 当实现所能解决的问题的最大规模达不到期望时决定改进还是放弃。

· 逐步改进实现,通过经验性分析或(和)数学分析验证改进后的效果。

· 用更高层次的抽象表示数据结构或算法来设计更高级的改进版本。

· 如果可能尽量为最坏情况下的性能提供保证,但在处理普通数据时也要有良好的性能。

· 在适当的时候将更细致的深入研究留给有经验的研究者并继续解决下一个问题。

算法(第4版)-1.5 案例研究:union-find算法的更多相关文章

  1. 《算法 (第4版)》【PDF】下载

    <算法 (第4版)>[PDF]下载链接: https://u253469.ctfile.com/fs/253469-231196349 (第4版)>[PDF]"  TITL ...

  2. 算法第四版-文字版-下载地址-Robert Sedgewick

    下载地址:https://download.csdn.net/download/moshenglv/10777447 算法第四版,文字版,可复制,方便copy代码 目录: 第1章 基 础 ...... ...

  3. 【原创】数据挖掘案例——ReliefF和K-means算法的医学应用

    数据挖掘方法的提出,让人们有能力最终认识数据的真正价值,即蕴藏在数据中的信息和知识.数据挖掘 (DataMiriing),指的是从大型数据库或数据仓库中提取人们感兴趣的知识,这些知识是隐含的.事先未知 ...

  4. Top100Summit全球案例研究峰会第一天总结——云计算和大数据

    很荣幸受邀参加Top100Summit全球软件案例研究峰会,这次的大会主题是<技术推动商业变革>,组委会从全国投稿的460多件案例中甄选出100件具有代表价值的案例,进行为期4天的分享,第 ...

  5. 十大经典排序算法的JS版

    前言 个人博客:Damonare的个人博客 如遇到问题或有更好的优化方法,可以: 提issue给我 或是pull requests 我都会看到并处理,欢迎Star. 这世界上总存在着那么一些看似相似但 ...

  6. Java虚拟机—垃圾回收算法(整理版)

    1.概述 由于垃圾收集算法的实现涉及大量的程序细节.因此本节不打算过多地讨论算法的实现,只是介绍几种算法的思想及其发展过程.主要涉及的算法有标记-清除算法.复制算法.标记-整理算法.分代收集算法. 2 ...

  7. Netty实战十四之案例研究(一)

    1.Droplr——构建移动服务 Bruno de Carvalho,首席架构师 在Droplr,我们在我的基础设施的核心部分.从我们的API服务器到辅助服务的各个部分都使用了Netty. 这是一个关 ...

  8. C#7.2——编写安全高效的C#代码 c# 中模拟一个模式匹配及匹配值抽取 走进 LINQ 的世界 移除Excel工作表密码保护小工具含C#源代码 腾讯QQ会员中心g_tk32算法【C#版】

    C#7.2——编写安全高效的C#代码 2018-11-07 18:59 by 沉睡的木木夕, 123 阅读, 0 评论, 收藏, 编辑 原文地址:https://docs.microsoft.com/ ...

  9. 洞察行业领先者的前沿思想——第五届TOP100全球软件案例研究峰会精彩谢幕

    (第五届TOP100summit开幕式现场) 12月09日-12日,由msup主办的第五届TOP100全球软件案例研究峰会(以下简称TOP100summit)在北京国家会议中心举行,作为互联网行业最有 ...

随机推荐

  1. Centos启动Cassandra交互模式失败:No appropriate python interpreter found

    在CentOS6.5安装好Cassandra后,启动交互模式: bin/sqlsh 192.168.10.154 时,报错 No appropriate python interpreter foun ...

  2. JS基本概念

    1.一切(变量.函数名.操作符)都区分大小写 2.标识符:第一个字符必须为字母.下划线或者美元符号,其他字符可以是字母.下划线.美元符号或者数字 3.数据类型 1)undefined:用var声明的变 ...

  3. 教你9个提升 Wordpress 网站安全性的方法

    大约一个月前,这个部落格被黑客入侵(编按:Amit Agarwal 的网站).而其他托管于相同主机商的网站像是 ctrlq.org 和2hundredzeros.com 也深受其害,黑客成功从网路上拿 ...

  4. css利用padding生成图标

    先插入HTML结构: <div class="line-three"></div> <div class="circle"> ...

  5. ios中属性和对象的初始化

    属性和对象的初始化为了方便记忆, 我们可以都使用self.来初始化. 这样可以避免内存的过度释放.

  6. OptionsMenu

    菜单是用户界面中最常见的元素之一,使用非常频繁,在Android中,菜单被分为如下三种,选项菜单(OptionsMenu).上下文菜单(ContextMenu)和子菜单(SubMenu),今天这讲是O ...

  7. DW安装步骤

    Adobe Dreamweaver CC 2015破解步骤 Adobe Dreamweaver CC 2015的安装包和破解补丁可以在这里下载.链接: http://pan.baidu.com/s/1 ...

  8. TortoiseGit流程安装使用手册

    end if

  9. MIUI5(红米、小米)打开开发者模式

    在miui5系统中系统默认隐藏原生android的开发者模式选项,要想启动该模式需要按照以下操作: 设置-关于手机- 连续点击安卓版本4下. 然后再返回主设置页面下,你会发现开发者选项已经出现.

  10. Leetcode 详解(Valid Number)

    Validate if a given string is numeric. Some examples:"0" => true" 0.1 " => ...