概要

可以通过对每个结点运行一次上一章中的最短路径算法,来解决所有结点对的最短路径问题,但效率较低,本章考虑的是用新的算法更高效的解决此问题

本章使用邻接矩阵来表示图,边的权重也存储在一个矩阵 W 中,算法的输出存储在矩阵 D 中,还有一个精妙的前驱结点矩阵用于存放路径结果

前驱结点矩阵的位置 ij 中存储的是,由 i 到 j 的一条最短路径中,j 的前驱结点

由前驱结点矩阵的第 i 行可以诱导出,由结点 i 出发的最短路径树

最短路径和矩阵乘法

这一节是使用动态规划的方法解决所有结点对的最短路径问题,按动态规划的设计方法,有四个步骤

步骤一:分析最优解结构

最优解必定是由子问题的最优解构成的,让本章的问题符合这一性质的是引理 24.1,一条最短路径的所有子路径都是最短路径

步骤二:所有结点对最短路径的递归解

这一段的分析过程基本没看明白,主要的目的是给出一个递归式,指明由子问题的解如何组成问题的解

公式给了两个,第一个公式 25.2 指出了如何由子问题的解组成问题的解,第二个公式 25.3 结合路径权重的场景具体分析,指出大于 n-1 条边的权重矩阵中的权重值都等于 n-1 条边的权重矩阵

步骤三:自底向上计算最短路径权重

计算的算法并不出奇,三层 for 循环遍历,利用公式 25.2 进行递归计算(递归应该算作第四次循环)

神奇的是这里计算的时候将公式 25.2 转成了矩阵乘法(没明白是如何转换的,数学功力不够),转成矩阵乘法之后算法的表述就变得更简单了,里面的三层 for 循环就是矩阵相乘的过程,外面的递归变成了不断的与同一个矩阵做乘法

步骤三改进:重复平方

结合路径权重的场景分析,我们只需要最终的权重矩阵,对计算的中间结果不感兴趣,利用 25.3 给出的性质,递归计算改为平方计算,即中间结果的矩阵与自己相乘,这样找出一个大于 n-1 阶的结果即可

这个改进可以将递归中的 O(n) 优化到 O(lgn)

步骤四:构建最优解

书中没有给出这一步骤,我思考就是利用概要中给出的方法,由前驱结点矩阵诱导出最短路径树

Floyd-Warshall 算法

Floyd-Warshall 算法的运行时间为 V 的三次方

Floyd-Warshall 算法也是基于动态规划的思想,但是是从另一个角度思考问题,下面的结构还是按照动态规划的思路组织的

最短路径结构(步骤一)

Floyd-Warshall 算法考虑的是一条最短路径上的中间结点(实话说从这句话里完全看不清算法的思路)

据我的理解,这里的问题和子问题分别是:

【问题】结点 i 到结点 j 的一条包含的中间结点全部取自 1 ~ k 号结点的最短路径

【子问题】结点 i 到结点 j 的一条包含的中间结点为全部取自 1 ~ k-1 号结点的最短路径

通俗一点讲就是,i 到 j 的路径的中间结点只能从 1 ~ k(或者 k-1)号结点中选择,所形成的最短路径

所有结点对最短路径问题的一个递归解(步骤二)

给出的递归解是这样的:关于问题 i 到 j 的最短路径,k 阶的 d 表示中间结点全部取自 1 ~ k 号结点的最短路径的权重,递归解说明了如何从 k-1 阶的 d 计算 k 阶的 d,最后计算到 n 阶的 d 就是在整个图中 i 到 j 的最短路径权重

自底向上计算最短路径权重(步骤三)

这一段比较好理解,按照递归式进行计算即可,算法非常紧凑,运行时间为 n 的三次方

构建一条最短路径(步骤四)

按照概述中的描述,构建最短路径需要通过前驱矩阵进行诱导(生成最短路径树)

那么问题就变成了如何构建前驱矩阵,书中说明了两种方法:

【方法一】由上述算法的计算结果 D 矩阵(由 n 阶的 d 组成的矩阵),来构造前驱矩阵

【方法二】在计算 D 矩阵的同时计算前驱矩阵,说是同时其实关联并不紧密,只是用到了 D 矩阵的中间结果,即 1 ~ n 阶的 D 矩阵

有向图的传递闭包

传递闭包是这样定义的:定义带权重的有向图 G,如果 G 中的结点 i 到 j 存在可以到达的路径,则(i,j)为 G 的传递闭包中的一条边

可以利用 Floyd-Warshall 算法计算图的传递闭包:给 G 的每条边赋予权重 1,运行 Floyd-Warshall 算法,结果矩阵 D 中小于 n 的 d 值就代表一条闭包中的边(或许书中的表述更易理解:如果存在一条 i 到 j 的路径,则有 d(ij)< n)

还有另外一种方法,用逻辑或 & 逻辑与操作替代 Floyd-Warshall 算法中的 min 和 + 操作(至于为什么可以这样替换我已经放弃思考了,应该是一个不太复杂的数学转换),这样改造后的算法运行会更快,因为逻辑或 & 逻辑与操作要比算法运算更快

我们可以利用 Floyd-Warshall 算法在 V 的三次方时间内计算出图的传递闭包

用于稀疏图的 Johnson 算法

Johnson 算法可以在 O(V²lgV+VE) 内找到所有结点对的最短路径,对于稀疏图,E 比较小,则 Johnson 算法性能优于本章的前两节的算法

Johnson 算法的思路是对每个结点运行一次 Dijkstra 算法

Dijkstra 算法要求权重都为非负值,所以这里可能需要(如果所有权重都为非负值则不需要)为每条边重新赋予一个非负值权重,再对带有新的权重的图运行 Dijkstra 算法。其中重新赋予权重要求,新的图的最短路径结果集与原图相同。

重新赋予权重来维持最短路径

这一小段讲的是如何计算一组与原图的最短路径相同的新权重

引理 25.1(重新赋予权重并不改变最短路径)中的公式 25.9 给出了新权重的计算方法,通过选择一个函数 h,利用公式 25.9 就能够计算出一组最短路径相同的新权重

通过重新赋值来生成非负权重

这一小段讲的是如何选择一个合适的函数 h,使得通过公式 25.9 计算出来的新权重都为非负值

选择 h 的方法为:向图 G 中添加一个新结点 s,定义 s 到其它所有结点的边的权重都为 0,然后计算由 s 结点出发的单源最短路径结果,函数 h(v) 定义为 s 到 v 的最短路径权重

书中同时给出了这样选择函数 h 的正确性的证明,利用三角不等式,简单明了

计算所有结点对之间的最短路径

这一小段给出了 Johnson 算法的具体过程:

先插入新结点 s 生成新的图 G‘ 并初始化

运行 Bellman-Ford 算法得到 s 的单元最短路径结果,构造函数 h

由函数 h 计算新的权重(非负值),形成新的权重矩阵

对每个结点运行 Dijkstra 算法(使用新的权重矩阵),得到一个结果矩阵 D’

由 D‘ 根据公式 25.9 反向计算出原图 G 的最短路径结果矩阵 D

此外,如果还需要构建最短路径,则可以在 D’ 的计算过程中保留前驱矩阵(将 Dijkstra 算法中生成前驱子图记录为前驱矩阵中的一行),这个前驱矩阵也可以作为原图 G (旧的权重矩阵)的前驱矩阵