Quick Find

直觉描述

有种问题叫动态联通性问题

动态联通性:简单

比如上图中的节点图,可以看到其中一部分被直接或间接连接起来了,比如:

(0, 1), (1, 2), (5, 6), (6, 7)

但有些没有,比如:

(2, 3), (0, 9)

如果现在我问你,这些节点中的某两个点是否连起来了?看起来不是个问题,除非问题变成如下图的形式。

动态联通性:复杂

这个问题就不那么容易回答了对吧?如果想自己试试,需要花不少时间。如果把每一个节点想象为一个计算机,我现在有100万台机器,问其中某几台是否连接,甚至都不大可能会是个可以手动解决的问题。

问题建模

首先,对于联通,需要知道其有三个特性:

  • 自连:(0, 0)是合法的连接;
  • 对称:如果(0, 1)存在,那么(1, 0)也存在;
  • 传递:如果(0, 1)和(1, 2)都存在,那么(0, 2)也存在;

有了这个前提,我们再来分解一下这个问题,转换为可解的题目。

已知条件:

数据的格式为一系列的节点,为了简便行事,我们就用index来标注它们。

[0, 1, 2, 3...N]

给出现有连接状态。如上面的简单示例,我们用union来描述两个节点之间曾经进行的连接行为。

union(0, 1), union(1, 2), union(0, 5), union(1, 6), 
union(2, 7), union(3, 4), union(3, 8), union(4, 9)

求:

对于已知的两个节点m, n,它们是否联通?
connected(m, n): Boolean

Quick Find

现在来想办法解决这个问题。Quick Find利用的是index。

针对每一个union,我们就用相同的index标注这两个节点(及其同组成员),直到结束。

  • 原数据index
    [0, 1, 2, ...N]
  • 连接
    union(0, 1)
  • 更新后的index
    [0, 0, 2, ...N]

那么比如说之前简单的联通性问题,可能在所有union完成后,index会更新为如下情形:
[0, 0, 0, 3, 3, 0, 0, 0, 3, 3]
之所以是可能,是因为我们在重新赋值index的时候,可能按照不同的原则,选择了不同的index来更新,比如也可以是如下的形式:
[1, 1, 1, 8, 8, 1, 1, 1, 1, 1]
这个需要特别在标注index时注意,union(m, n)意味着m和n所对应的阵营合并了,而不仅仅是节点本身。

完成标注后,如果我们想知道09这两个节点是否连接,我们可以:
connected(0, 9)
实现的形式为:
查询0和9的index,看其是否相等,如果是则返回true,否则false

复杂度

这种方法做了如下步骤:

  • 建立index并初始化
  • 记录每个union:
  • 查询联通性

在这里,我们暂且将复杂度这个概念简单等同为访问了index的次数。结合如下代码,我们来看看这个算法需要多少复杂度。

public class QuickFindUF
{
    private int[] id;
    public QuickFindUF(int N)
    {
        id = new int[N];
        for (int i = 0; i < N; i++)
          id[i] = i;
    }
    public boolean connected(int p, int q)
    { return id[p] == id[q]; }
    public void union(int p, int q)
    {
        int pid = id[p];
        int qid = id[q];
        for (int i = 0; i < id.length; i++)
          if (id[i] == pid) id[i] = qid;
    }
}

建立index并初始化

用到的是constructor中的for循环,即:

for (int i = 0; i < N; i++)
    id[i] = i;

很明显,这里访问了index对象N次。

记录每个union

这里用到的是union方法,该方法也有一个for循环。

最坏的情况,比如对于[0, 0, 0, 0, ...0]这样的index对象,union(0, 8)会遍历index中的每一个元素,并且改写其id,所以分解该方法的每一步,我们可以知道:

int pid = id[p]; //1次
int qid = id[q]; //1次
for (int i = 0; i < id.length; i++) //N次
    if (id[i] == pid) //1次
      id[i] = qid; //1次

总共访问了1 + 1 + N * (1 + 1) = 2 + 2N次。

查询联通性

这里用到的是connected方法:

public boolean connected(int p, int q)
    { return id[p] == id[q]; }

可以看到,==两端各一次访问,因此总次数是2次。

是否经济

简单做一下分析,如果我们有N个节点,其中预设有M个连接,现在要查询P组节点的联通性情况,我们需要多少计算量?

  • 初始化:N
  • union:N*M
  • 查询:P

初始化和查询都还好,问题出在union这里。简单期间,假设M = N。N2意味着什么?

  • 1000个节点需要1,000,000计算量;
  • 100,000个节点需要10,000,000,000计算量;

计算量以指数形式上升,听起来就不是件好事。到底怎么不好?我们换个方式说:

  • 当代计算机的能力大概是每秒109次的计算(访问数据的)能力;
  • 如果我们在内存有109个单词要处理(访问);
  • 处理时间为1秒;
  • 随着内存不断增大,如果计算能力也线性增大,那么处理时间会保持不变;
  • 1019 内存匹配 1019运算力的CPU,满内存运算依旧是1秒;

好,那么对于Quick Find的问题:

  • 假设N=109,我们需要N2的访问;
  • 计算机每秒109的访问/计算能力以及内存;
  • 所需时间为109秒 = 30年;
  • 如果计算机能力同上升级,1019内存匹配1019运算力的CPU,满内存运算会变为1019秒 = 3000亿年左右。这里比较神经的一点是,我计算机能力提高了1010倍的前提下,满内存运算居然慢了这么多??

指数膨胀的算法,确实不好,是因为它无法规模化,规模越大,效率越低

用个很恶心的例子,如果我养猪,养1头需要100斤饲料/年,养10头需要1000斤是合理的;但如果有人告诉你养10头需要10,000斤饲料,养100头需要1,000,000,这样的只怕没人养猪,或者让猪改吃垃圾更合理?

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,948评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,371评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,490评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,521评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,627评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,842评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,997评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,741评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,203评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,534评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,673评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,339评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,955评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,770评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,000评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,394评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,562评论 2 349

推荐阅读更多精彩内容

  • 本系列博客习题来自《算法(第四版)》,算是本人的读书笔记,如果有人在读这本书的,欢迎大家多多交流。为了方便讨论,本...
    kyson老师阅读 1,561评论 0 47
  • 本系列博客习题来自《算法(第四版)》,算是本人的读书笔记,如果有人在读这本书的,欢迎大家多多交流。为了方便讨论,本...
    kyson老师阅读 2,222评论 0 50
  • 夜将至未至 整个城市都枯了 除了灯红酒绿 除了流离失所 除了无所事事的我 我走遍了所有大街小巷 记得垃圾桶的位置 ...
    大叔的小黑屋阅读 439评论 2 2
  • 他双手拖着一把漆黑之色的巨剑向这边走来,似是很沉重,可他却也欢喜的出来。 他拖着这把与现世格格不入的物件,在左右两...
    chajn阅读 487评论 0 2
  • 11岁的儿子在厕所里洗澡,大叫着说:“老爸,我浴巾忘拿了,拜托帮我拿一下!” 老公把浴巾拿到门口,刚从虚掩的门瞅了...
    c小尘阅读 637评论 2 1