Back to Oi Wiki

Dsu

docs/ds/dsu.md

latest17.0 KB
Original Source

author: HeRaNO, JuicyMio, Xeonacid, sailordiary, ouuan, Pig-Eat-Earth

引入

并查集是一种用于管理元素所属集合的数据结构,实现为一个森林,其中每棵树表示一个集合,树中的节点表示对应集合中的元素.

顾名思义,并查集支持两种操作:

  • 合并(Unite):合并两个元素所属集合(合并对应的树).
  • 查询(Find):查询某个元素所属集合(查询对应的树的根节点),这可以用于判断两个元素是否属于同一集合.

并查集在经过修改后可以支持单个元素的删除、移动或维护树上的边权.使用动态开点线段树还可以实现 可持久化并查集

???+ warning "Warning" 并查集无法以较低复杂度实现集合的分离.

初始化

初始时,每个元素都位于一个单独的集合,表示为一棵只有根节点的树.方便起见,我们将根节点的父亲设为自己.

???+ example "实现" === "C++" ```cpp struct dsu { vector<size_t> pa;

      explicit dsu(size_t size) : pa(size) { iota(pa.begin(), pa.end(), 0); }
    };
    ```

=== "Python"
    ```python
    class Dsu:
        def __init__(self, size):
            self.pa = list(range(size))
    ```

查询

我们需要沿着树向上移动,直至找到根节点.

???+ example "实现" === "C++" cpp size_t dsu::find(size_t x) { return pa[x] == x ? x : find(pa[x]); }

=== "Python"
    ```python
    def find(self, x):
        return x if self.pa[x] == x else self.find(self.pa[x])
    ```

路径压缩

查询过程中经过的每个元素都属于该集合,我们可以将其直接连到根节点以加快后续查询.

???+ example "实现" === "C++" cpp size_t dsu::find(size_t x) { return pa[x] == x ? x : pa[x] = find(pa[x]); }

=== "Python"
    ```python
    def find(self, x):
        if self.pa[x] != x:
            self.pa[x] = self.find(self.pa[x])
        return self.pa[x]
    ```

合并

要合并两棵树,我们只需要将一棵树的根节点连到另一棵树的根节点.

???+ example "实现" === "C++" cpp void dsu::unite(size_t x, size_t y) { pa[find(x)] = find(y); }

=== "Python"
    ```python
    def unite(self, x, y):
        self.pa[self.find(x)] = self.find(y)
    ```

启发式合并

合并时,选择哪棵树的根节点作为新树的根节点会影响未来操作的复杂度.我们可以将节点较少或深度较小的树连到另一棵,以免发生退化.

??? note "具体复杂度讨论" 由于需要我们支持的只有集合的合并、查询操作,当我们需要将两个集合合二为一时,无论将哪一个集合连接到另一个集合的下面,都能得到正确的结果.但不同的连接方法存在时间复杂度的差异.具体来说,如果我们将一棵点数与深度都较小的集合树连接到一棵更大的集合树下,显然相比于另一种连接方案,接下来执行查找操作的用时更小(也会带来更优的最坏时间复杂度).

当然,我们不总能遇到恰好如上所述的集合——点数与深度都更小.鉴于点数与深度这两个特征都很容易维护,我们常常从中择一,作为估价函数.而无论选择哪一个,时间复杂度都为 $O (m\alpha(m,n))$,具体的证明可参见 References 中引用的论文.

在算法竞赛的实际代码中,即便不使用启发式合并,代码也往往能够在规定时间内完成任务.在 Tarjan 的论文[^tarjan1984worst]中,证明了不使用启发式合并、只使用路径压缩的最坏时间复杂度是 $O (m \log n)$.在姚期智的论文[^yao1985expected]中,证明了不使用启发式合并、只使用路径压缩,在平均情况下,时间复杂度依然是 $O (m\alpha(m,n))$.

如果只使用启发式合并,而不使用路径压缩,时间复杂度为 $O(m\log n)$.由于路径压缩单次合并可能造成大量修改,有时路径压缩并不适合使用.例如,在可持久化并查集、线段树分治 + 并查集中,一般使用只启发式合并的并查集.

按节点数合并的参考实现:(注意需要调整初始化方法)

???+ example "实现" === "C++" ```cpp struct dsu { vector<size_t> pa, size;

      explicit dsu(size_t size_) : pa(size_), size(size_, 1) {
        iota(pa.begin(), pa.end(), 0);
      }
    
      void unite(size_t x, size_t y) {
        x = find(x), y = find(y);
        if (x == y) return;
        if (size[x] < size[y]) swap(x, y);
        pa[y] = x;
        size[x] += size[y];
      }
    };
    ```

=== "Python"
    ```python
    class Dsu:
        def __init__(self, size):
            self.pa = list(range(size))
            self.size = [1] * size
    
        def unite(self, x, y):
            x, y = self.find(x), self.find(y)
            if x == y:
                return
            if self.size[x] < self.size[y]:
                x, y = y, x
            self.pa[y] = x
            self.size[x] += self.size[y]
    ```

参考实现

带有路径压缩、按节点数合并的并查集的完整实现如下所示:

??? example "模板题 Luogu P3367【模板】并查集 参考实现" === "C++" cpp --8<-- "docs/ds/code/dsu/dsu_0.cpp"

=== "Python"
    ```python
    --8<-- "docs/ds/code/dsu/dsu_0.py"
    ```

复杂度

同时使用路径压缩和启发式合并之后,并查集的每个操作平均时间仅为 $O(\alpha(n))$.其中,$\alpha$ 为阿克曼函数的反函数,增长极其缓慢.也就是说,并查集单次操作的平均运行时间可以认为是一个很小的常数.时间复杂度的证明在 这个页面 中.

???+ info "反 Ackermann 函数" Ackermann 函数 $A(m, n)$ 的定义是这样的:

$A(m, n) = \begin{cases}n+1&\text{if }m=0\\A(m-1,1)&\text{if }m>0\text{ and }n=0\\A(m-1,A(m,n-1))&\text{otherwise}\end{cases}$

而反 Ackermann 函数 $\alpha(n)$ 的定义是 Ackermann 函数的反函数,即为最大的整数 $m$ 使得 $A(m, m) \leqslant n$.

并查集的空间复杂度显然为 $O(n)$.

拓展操作

在普通的并查集的基础上,还可以做一系列修改使之支持更多的操作或维护更复杂的信息.

带删除并查集

普通的并查集无法支持删除操作,是因为删除一个节点的时候,不可避免地会将以它为根的子树上所有节点都删除.为了解决这一问题,在带删除操作的并查集中,可以通过建立虚点的方法保证所有实际存储数据的节点总是叶子节点.为此,需要在初始化时,就为每个数据节点都建立一个虚点,并将数据节点的父节点设置为该虚点.由于每次合并两个集合时,都只会将两个集合的树根连接,所以,从始至终只有虚点会有子节点.这就保证了删除一个节点时,不会误删其他节点.

注意,删除单个节点后,需要重新为该节点建立一个虚点作为其父节点;否则,无法正确执行后续的合并和删除操作.

??? example "模板题 SPOJ JMFILTER - Junk-Mail Filter 参考实现" === "C++" cpp --8<-- "docs/ds/code/dsu/dsu_4.cpp"

=== "Python"
    ```python
    --8<-- "docs/ds/code/dsu/dsu_4.py"
    ```

类似的方法还可以用于实现在集合间移动单个元素.实现细节详见例题.

带权并查集

我们还可以在并查集的边上定义某种权值和这种权值在路径压缩时产生的运算,从而解决更多的问题.比如对于经典的「NOI2001」食物链,我们可以在边权上维护模 $3$ 意义下的加法群.对于这类维护模意义下边权且模数很小的问题,还可以通过将并查集的单个点拆分为多个状态的方式来解决.这种特殊情形下的技巧,也称为「种类并查集」或「拓展域并查集」.后文会通过例题来说明这些做法.

为了维护并查集中的边权,需要将边权下放到子节点中存储.因此,每个节点存储的都是它到它的父节点之间的边权.只有当一个节点的父节点发生变化时,才需要相应地调整边权.一般情形中,这可能发生在路径压缩和合并两个节点时.例如,如果边权是当前节点与父节点之间的距离,那么,在路径压缩时,每次将当前节点的父节点替换为根节点,都需要将父节点到根节点的距离加到当前节点存储的边权上;类似地,在合并两个节点所在集合时,需要计算两个根节点之间新连接的边的权值.

??? example "模板题 Library Checker - Unionfind with Potential 参考实现" === "C++" cpp --8<-- "docs/ds/code/dsu/dsu_5.cpp"

=== "Python"
    ```python
    --8<-- "docs/ds/code/dsu/dsu_5.py"
    ```

例题

算法竞赛中,直接考察并查集的题目大多都需要针对题目设计特殊的结构.

???+ example "UVa11987 Almost Union-Find" 实现类似并查集的数据结构,支持以下操作:

1.  合并两个元素所属集合.
2.  将单个元素移动到另一个元素所在的集合.
3.  查询某个元素所属集合的大小及元素和.

??? note "解答" 这道题目中,操作 1 和操作 3 都容易处理,难点在于操作 2.假定要将元素 $x$ 移动到元素 $y$ 所在的集合.在普通的并查集中,直接将元素 $x$ 的父亲设为元素 $y$ 所在集合的根节点是不行的,因为这样会将元素 $x$ 所在子树的元素都一起移动.针对这个问题,解决方法就是保证元素 $x$ 没有子节点.为此,在建立并查集时为每个元素 $x$ 都建立一个虚点 $\tilde x$,并将元素 $x$ 的父亲指向对应的虚点 $\tilde x$.这样,在合并两个集合的时候,因为总是将一个树根连接到另一个树根,而树根又全部是虚点,所以,只有虚点会有子节点,而所有实际存储元素的点都没有子节点.此时,要移动元素,就容易实现得多.

??? note "参考实现" === "C++" cpp --8<-- "docs/ds/code/dsu/dsu_1.cpp"

=== "Python"
    ```python
    --8<-- "docs/ds/code/dsu/dsu_1.py"
    ```

???+ example "Luogu P2024「NOI2011」食物链" 动物王国中有三类动物 $A,B,C$,这三类动物的食物链构成了有趣的环形.$A$ 吃 $B$,$B$ 吃 $C$,$C$ 吃 $A$.

现有 $N$ 个动物,以 $1 \sim N$ 编号.每个动物都是 $A,B,C$ 中的一种,但是我们并不知道它到底是哪一种.

有人用两种说法对这 $N$ 个动物所构成的食物链关系进行描述:

-   第一种说法是 `1 X Y`,表示 $X$ 和 $Y$ 是同类.
-   第二种说法是 `2 X Y`,表示 $X$ 吃 $Y$.

此人对 $N$ 个动物,用上述两种说法,一句接一句地说出 $K$ 句话,这 $K$ 句话有的是真的,有的是假的.当一句话满足下列三条之一时,这句话就是假话,否则就是真话.

-   当前的话与前面的某些真的话冲突,就是假话;
-   当前的话中 $X$ 或 $Y$ 比 $N$ 大,就是假话;
-   当前的话表示 $X$ 吃 $X$,就是假话.

你的任务是根据给定的 $N$ 和 $K$ 句话,输出假话的总数.

??? note "解答一" 考虑用带权并查集维护食物链信息.如果 $x$ 和 $y$ 是同类,那么 $x\equiv y\pmod 3$;如果 $x$ 吃 $y$,那么 $x - y \equiv 1 \pmod 3$.这样就将本题转化为前文的模板题.

具体地,对于每一句话,除去那些那些 $x>n$ 或 $y>n$ 的显然的假话外,需要判断 $x$ 和 $y$ 是否已经连接:如果已经连接,计算两者的模意义下的距离,并与这句话声称的信息进行比较;否则,将两者按照这句话提供的信息连接.除了显然的情形外,一句话是假话,当且仅当提到的两个节点已经连接,且对应的距离与这句话声称的信息矛盾.

??? note "参考实现一" === "C++" cpp --8<-- "docs/ds/code/dsu/dsu_6.cpp"

=== "Python"
    ```python
    --8<-- "docs/ds/code/dsu/dsu_6.py"
    ```

??? note "解答二" 将一种生物 $x$ 拆分为三种状态.在具体实现中,我们可以直接将不同的状态当作不同的元素:

-   与 $x$ 处于同一集合的状态与 $x$ 属于同一物种;
-   与 $x+n$ 处于同一集合的状态能被 $x$ 吃;
-   与 $x+2n$ 处于同一集合的能吃 $x$.

于是,对于一句话:

-   `1 x y` 为假话当且仅当:

    1.  $x>N$ 或 $y>N$;
    2.  $y$ 与 $x+n$ 或 $x+2n$ 中的一个处于同一集合内.
-   `2 x y` 为假话当且仅当:

    1.  $x>N$ 或 $y>N$;
    2.  $y$ 与 $x$ 或 $x+2n$ 中的一个处于同一集合内.
-   若为真话,合并对应状态.

??? note "参考实现二" === "C++" cpp --8<-- "docs/ds/code/dsu/dsu_2.cpp"

=== "Python"
    ```python
    --8<-- "docs/ds/code/dsu/dsu_2.py"
    ```

???+ example "ABC396E Min of Restricted Sum" 给定整数 $N, M$ 和长度为 $M$ 的整数序列 $X=(X_1,X_2,\ldots,X_M)$、$Y=(Y_1,Y_2,\ldots,Y_M)$、$Z=(Z_1,Z_2,\ldots,Z_M)$.其中,保证 $X$ 和 $Y$ 的所有元素均在 $1$ 至 $N$ 的范围内.

定义长度为 $N$ 的非负整数序列 $A=(A_1,A_2,\ldots,A_N)$ 为 **好的整数序列**,当且仅当满足以下条件:

-   对于所有满足 $1 \leq i \leq M$ 的整数 $i$,有 $A_{X_i} \oplus A_{Y_i} = Z_i$,其中 $\oplus$ 表示异或运算.

请判断是否存在这样的好的整数序列.若存在,请找出使得元素总和 $\displaystyle \sum_{i=1}^N A_i$ 最小的好的整数序列,并输出该序列.

??? note "解答" 异或就是单个二进制位上的「相同」或「不同」关系.那么,将 $A_i$ 的所有二进制位拆开,异或关系就能用带权并查集(或种类并查集)维护了.同一个连通块内的元素一定对应着 $A$ 中不同数字的同一个数位.统计答案时,同一连通块的元素通常分为两组,两组之间取值应当不同,只需要取其中较大的一组赋值为 $0$,另一组赋值为 $1$ 即可保证总权值最小.

??? note "参考实现" === "C++" cpp --8<-- "docs/ds/code/dsu/dsu_3.cpp"

=== "Python"
    ```python
    --8<-- "docs/ds/code/dsu/dsu_3.py"
    ```

习题

其他应用

最小生成树算法 中的 Kruskal 和 最近公共祖先 中的 Tarjan 算法是基于并查集的算法.

相关专题见 并查集应用

参考资料与拓展阅读

  1. 知乎回答:是否在并查集中真的有二分路径压缩优化?
  2. Gabow, H. N., & Tarjan, R. E. (1985). A Linear-Time Algorithm for a Special Case of Disjoint Set Union. JOURNAL OF COMPUTER AND SYSTEM SCIENCES, 30, 209-221.PDF
  3. CSDN:扩展域并查集 & 带权并查集