并查集(Union Find):实现及其优化(c++)

1.什么是并查集

并查集是用来管理元素分组的数据结构。可以高效进行如下操作:

  • 查询元素a、b十是否在同一组
  • 合并a、b所在的组

并查集可以进行合并操作但不能进行分割操作。

2.并查集的结构

并查集采用多叉树形结构实现,每个元素对应一个结点,每个组对应一棵树。重点关注结整体形成一个树形结构,而不是树的形状等信息。


3.并查集的实现

3.1 初始化

对于并查集,一般采用数组来实现,其中元素为数组的索引,其父辈为数组索引对应内容。
在初始化中,将每个元素父辈设为自己,即自己形成一组,并对用一个rank数组记录以每个元素为根的树的层数,以方便后面并查集优化的实现。


    UnionFind(int count)
    {
        parent = new int[count];
        rank = new int[count];
        this->count = count;
        for (int i = 0; i < count; ++i)
        {
            parent[i] = i;
            rank[i] = 1;
        }
    }
1 2 3 4 5
parent 1 2 3 4 5

3.2 查找

为了查询两个节点否属于同一数组,需要采用查找两个节点的根,通过判断根是否相同来判断元素是否相同的方法。故查找用于实现查找元素的根。

在查找过程中采用路径压缩对组内进行优化,将树的层数降低,从而降低查找的复杂度,通常有两种压缩方法:

  • 采用递归实现,在查询过程中向上经过的所有节点都直接连到根上,将路径压缩至如下状态,使得查找时间复杂度降为 O(1),但路径压缩的过程则比较耗费时间。


    int find(int p)
    {
        assert(p >= 0 && p < count);
        /*****  路径压缩一:压缩过程耗时 *****/
        if(p != parent[p])
            parent[p] = find(parent[p]);        //递归实现路径压缩,最终find时间复杂度为 O(1)
        return parent[p];
    }
  • 采用循环,在查询过程中,将当前结点的父辈更改为当前父辈的父辈,则将路径压缩至如下状态,查找时间复杂度虽然未达到最优,但压缩时间开销相对较小。


    int find(int p)
    {
        assert(p >= 0 && p < count);
        /*****  路径压缩二:压缩过程快,find时间复杂度未达到 O(1)   *****/
        while (p != parent[p])
        {
            parent[p] = parent[parent[p]];
            p = parent[p];
        }
        return p;
    }

3.3 合并

将一组元素的根指向另一组元素的根,就实现了元素的合并:

  • 因为只要属于两个组的元素进行了合并,就相当于这两个组合并了。故两元素合并时先找到其所在组的根,再把根合并。
  • 在合并过程中,需要将层数低的数指向层数高的,以有效降低合并后树的高度,这时候就用到了我们初始化时使用的 rank 数组。
    void unionElements(int p, int q)
    {   
        //合并优化,降低层数
        
        int pRoot = find(p);
        int qRoot = find(q);
        if(pRoot == qRoot)
            return;
        if(rank[pRoot] > rank[qRoot])
        {
            parent[qRoot] = pRoot;
        }
        else if(rank[pRoot] < rank[qRoot])
        {
            parent[pRoot] = qRoot;
        }
        else
        {   //rank[pRoot] == rank[qRoot]的情况
            parent[qRoot] = pRoot;
            rank[pRoot] += 1;
        }        
    }

3.4 判断两元素是否如同组

判断元素的根是否相同,具体实现如下:

    bool isconnected(int p, int q)
    {
        return find(p) == find(q);
    }

4.小结

  • 通过优化后的并查集效率非常高。对于 n 个元素进行一次操作的复杂度时 O(a(n))。 a(n)时阿克曼(Ackermann)函数的反函数,比 O(log(n)) 还快。
  • 在最小生成树算法 Kruskal 中,需要判断一条边的两个顶点是否属于同一个连通分量,这时候需要并查集来进行高效的判断。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,258评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,335评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,225评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,126评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,140评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,098评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,018评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,857评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,298评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,518评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,678评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,400评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,993评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,638评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,801评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,661评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,558评论 2 352

推荐阅读更多精彩内容

  • “并查集”这部分知识点讲得最清楚的是《算法》(第 4 版),本篇“并查集”的介绍是我看这本书第 1.5 节的学习笔...
    李威威阅读 871评论 0 2
  • 并查集(Union Find) 需求分析 假设现在有这样一个需求,如下图的每一个点代表一个村庄,每一条线就代表一条...
    ducktobey阅读 1,402评论 0 1
  • 一、基本概念和定义 参考文章并查集(Union-find Sets)是一种非常精巧而实用的数据结构,它主要用于处理...
    一只可爱的柠檬树阅读 613评论 0 0
  • 前言 并查集(Disjoint-set) 的代码非常简洁,但是功能却很强大。关于并查集,这里有一篇文章超有爱的并查...
    程点阅读 23,196评论 1 20
  • 新型“家长会” 今天家长会以一种新型的方式展开,分组讨论,围绕主题,这对于我们家长来说是有益处的,可见校方也是用心...
    Hailey27阅读 107评论 0 0