WenjieWei Blog
open main menu
Part of series:Algorithm

传递闭包概念及其算法

/ 4 min read

定义

一个nn个顶点的有向图的传递闭包可以定义为一个nn阶的布尔矩阵T={tij}T=\{t_{ij}\},其中tij=1t_{ij}=1当且仅当从顶点ii到顶点jj有一条有效的有向路径,否则tij=0t_{ij}=0

例子

给定如下图的有向图:

Loading diagram...
Source
graph LR
    A --> B
    B --> D
    D --> A
    D --> C

那么其邻接矩阵为:

A=[0100000100001010]\begin{equation} A = \begin{bmatrix} 0 & 1 & 0 & 0 \\ 0 & 0 & 0 & 1 \\ 0 & 0 & 0 & 0 \\ 1 & 0 & 1 & 0 \end{bmatrix} \end{equation}

其传递闭包为:

R=[1111111100001111]\begin{equation} R = \begin{bmatrix} 1 & 1 & 1 & 1 \\ 1 & 1 & 1 & 1 \\ 0 & 0 & 0 & 0 \\ 1 & 1 & 1 & 1 \end{bmatrix} \end{equation}

通俗来讲,邻接矩阵表征的是图中两个顶点之间是否有边,而传递闭包表征的是两个顶点之间的传递关系。

Floyd-Warshall 算法

该算法用于求解任意两点之间的最短路径,同时也用于计算有向图的传递闭包,本次只记录后者。

时间复杂度为O(n3)O(n^3), 空间复杂度为O(n2)O(n^2)

算法描述

该算法是一个经典的动态规划算法:

对于求最短路径:求解的问题是任意节点ii到节点jj的最短路径。该问题可以分解为如下子问题:我们假设节点kkiijj路径上的中间节点,那么iijj的路径可以分解为iikk的路径和kkjj的路径,按此思路遍历所有节点kk,即可求得iijj的最短路径。

对于求传递闭包:求解的问题是第一个节点到节点nn是否有路径。同样的,第kk个节点的子问题是节点11到节点kk的传递闭包。

算法过程

还是以该有向图为例:

Loading diagram...
Source
graph LR
    A --> B
    B --> D
    D --> A
    D --> C

过程如图所示:

marshall_ts

  1. 初始化传递闭包矩阵R0R0为邻接矩阵AA
  2. 对于第一个节点,添加了第一行第一列,有找到[d,a]=1[d,a]=1, 即 D 和 A 之间存在路径,也就是说只要 A 能达到的节点,D 也能达到; 又有[a,b]=1[a,b]=1, 所以[d,b]=1[d,b]=1, 所以R1R1中更新值 (用绿色表示)
  3. 对于第二个节点,添加了第二行第二列,由[a,b]=1[a,b]=1[d,b]=1[d,b]=1,以及[b,d]=1[b,d]=1, 所以[a,d]=1[a,d]=1, [d,d]=1[d,d]=1, R2R2中更新值
  4. 对于第三个节点,添加了第三行第三列,但是行方向上的值都为 0, 也就是说这个节点无法到达其他节点,所以R3R3中无需更新值
  5. 对于第四个节点,添加了第四行第四列,由[a,d]=1[a,d]=1 [b,d]=1[b,d]=1 [d,d]=1[d,d]=1, 并且列方向上的值都为 1, 也就是说这个节点可以到达其他节点,因此aa行和bb行都为 1,R4R4中更新值
  6. 最终得到最后的传递闭包矩阵RR

简单实现

fn transitive_closure(adj: Vec<Vec<i32>>) -> Vec<Vec<i32>> {
    let n = adj.len();
    let mut res = adj.clone();
    for k in 0..n {
        for i in 0..n {
            for j in 0..n {
                res[i][j] = res[i][j] | (res[i][k] & res[k][j]);
            }
        }
    }
    res
}

代码实例:https://play.rust-lang.org/?version=stable&mode=debug&edition=2021&gist=65dfec209cc63ebe2e6cb95c45fb205b