引言:Floyd-Warshall算法作为经典的动态规划算法,能够在O(n3)复杂度之内计算出所有点对之间的最短路径,且由于其常数较小,对于中等规模数据运行效率依然可观。算法共使用n此迭代,n为顶点个数。其中第k次迭代计算出每对顶点之间所有中间结点小于等于k的最短路径长度,其中i到j的最短路径要么是经过k的一条路径,这条路径的由k所分割出的两个路径i → k、k → j是中间路径小于等于k-1的最短路径;要么是从i到j的中间路径小于等于k-1的最短路径。定义dij(k)为从i到j的最短路径长度,可以形式化地表示其计算方法:

(1)

算法为了得出最短路径的所有中间结点,维护前趋矩阵Π。其元素 πij 表示从i到j的最短路径上j前一结点,其计算方法如下:

                 (2)

         (3)

从前趋矩阵得出最短路径所有结点的方法如下:

PRINT-SHORTEST-PATH (i, j, Π)

    1   STACK ← Ø

    2   while j ≠ i

    3      PUSH (STACK, j)

    4       j ← Π (i, j)

    5   PUSH (STACK, j)

    6   while STACK ≠ Ø

    7       v ← POP (STACK)

    8       print v

问题描述:关于Floyd-Warshall算法的细节请参见《算法导论》或其他算法专著,笔者在阅读《算法导论》相关章节的时候萌生了关于题目所述问题的猜想,表述如下:算法最终得出dij之前,一定在之前的某个第k次迭代对dij(k)做了使其等于dik(k-1) dkj(k-1)的赋值,假设k为最后这一次这样的迭代,那么算法隐含了这样一个信息:最终从i到j的最短路径中间经过k,且从i到k和从k到j的中间路径小于等于k-1。而从i到k和从k到j亦符合类似的模式,这样一直递归下去,直到得到的两个点的最短路径为图中一条边。可以为所有最短路径构造一棵树表示这个过程。举例来说,对于路径3257416,其相对应的反映了算法构造路径过程的树为:

图1

那么,由PRINT-SHORTEST-PATH所得出的路径是不是就这样一条路径呢?答案竟然是肯定的。下面将会给出证明,其中,推论1的证明使用了最短路径的一些性质,而结论的证明使用的就是算法证明里最重要的递归思想和数学归纳法。

首先给出并证明一个比较重要的推论。

推论1若在第k次迭代,某个从i到j的路径经过了k,令m = πij(k),如果m ≠ k,则从i到m的路径也经过k。

证明:由于πkj(k-1) = πij(k)= m,即,从k到j的中间结点小于等于k-1在最短路径中,j的前趋为m,所以有

                                dkj(k-1) = dkm(k-1) + wmj                                                            (4,最短路径的最优子结构性质)

其中wmj为边(m, j)的权值。

因为从m到j有一条边,由最短路径的三角不等式可得:

dij(k-1) ≤ dim(k-1) wmj                                                              (反证即可得)

dim(k-1)  ≥  dij(k-1) wmj                                                             (5)

由条件可知

                                dij(k-1) dik(k-1) dkj(k-1)

两边减去wmj,得

                                dij(k-1)  - wmj  dik(k-1) dkj(k-1)  wmj =  dik(k-1) dkm(k-1)          (由4)

dij(k-1)  - wmj  dik(k-1) dkm(k-1) 

得证

结论对由PRINT-SHORTEST-PATH所得出的路径构造一颗如图1所示的树,这颗树反映了Floyd-Warshall算法构造最短路径的过程。

证明:利用数学归纳法,显而易见,第一次迭代之前,对初始的前趋矩阵应用PRINT-SHORTEST-PATH函数得到的路径反映了算法初始时没有中间结点的最短路径的构造过程。

接下来,假设,算法对第k-1次迭代所得的前趋矩阵Πk-1,应用PRINT-SHORTEST-PATH函数得到的路径,反映了算法构造中间结点小于等于k-1的最短路径的过程。并根据这个假设,证明算法对第k次迭代所得的前趋矩阵Πk,应用PRINT-SHORTEST-PATH函数得到的路径,反映了算法构造中间结点小于等于k的最短路径的过程即可。

1. 对于某个从i到j的算法所构造出的中间结点小于等于k的最短路径,若其不经过k,有 dij(k) = dij(k-1),πij(k) = πij(k-1) = m,对其应用PRINT-SHORTEST-PATH函数可得序列:

πij(k) = m

πim(k) = n

...

πio(k) = p

πip(k) = i

对于其中任意一个 πix(k),其一定等于πix(k-1),即从i到这条路径中任意一个x的最短路径都不会经过k,否则就会与假设矛盾。因此有:

πij(k) = πij(k-1) = m

πim(k) = πim(k-1) = n

...

πio(k) = πio(k-1) = p

πip(k) = πip(k-1) = i

即根据Πk所计算出来的从i到j的路径等同于根据Πk-1所计算出来的从i到j的路径。由于根据Πk-1所计算出来的路径是符合性质的中间结点小于等于k-1的路径,因此根据Πk,对任意不经过k的最短路径应用PRINT-SHORTEST-PATH函数计算出的路径即是符合性质的中间结点小于等于k的路径。

2. 若从i到j的路径经过k,即

           dij(k) = dik(k-1) dkj(k-1),πij(k) = πkj(k-1)= m

由推论1可得,根据Πk,对从i到j的最短路径应用PRINT-SHORTEST-PATH函数计算出的路径一定会到达k,且有:

πij(k) = πkj(k-1) = m

πim(k) = πkm(k-1) = n

...

πio(k) = πko(k-1) = p

πip(k) = πkp(k-1) = k

这个从k到j的序列的所有点都来自Πk-1,所以它是符合性质的中间结点小于等于k-1的路径。

而对于从i到k的序列,由于其不经过k,因此根据1,也是符合性质的中间结点小于等于k-1的路径。

由于路径从k一分为二且k为所有中间结点中最大的那个,所以这条路径符合所构造的树先从最大的中间结点分开的性质。而分开的两条路径又分别是符合性质的中间结点小于等于k-1的路径,因此第k迭代构造的经过k的路径是符合性质的中间结点小于等于k的路径。

综上所述,可以得出结论

关于Floyd-Warshall算法由前趋矩阵计算出的最短路径反映出了算法的执行过程特性的证明...相关推荐

  1. Floyd Warshall算法

    Description: 描述: This is a very popular interview problem to find all pair shortest paths in any gra ...

  2. C++floyd warshall算法求最短路径(附完整源码)

    C++floyd warshall算法求最短路径 floyd warshall算法求最短路径的完整源码(定义,实现,main函数测试) floyd warshall算法求最短路径的完整源码(定义,实现 ...

  3. Floyd - Warshall(弗洛伊德算法)

    简介:Floyd算法又称为插点法,是一种利用动态规划的思想寻找给定的加权图中多源点之间最短路径的算法,与Dijkstra算法类似.该算法名称以创始人之一.1978年图灵奖获得者.斯坦福大学计算机科学系 ...

  4. 常用十大算法 非递归二分查找、分治法、动态规划、贪心算法、回溯算法(骑士周游为例)、KMP、最小生成树算法:Prim、Kruskal、最短路径算法:Dijkstra、Floyd。

    十大算法 学完数据结构该学什么?当然是来巩固算法,下面介绍了十中比较常用的算法,希望能帮到大家. 包括:非递归二分查找.分治法.动态规划.贪心算法.回溯算法(骑士周游为例).KMP.最小生成树算法:P ...

  5. c语言知道算法写不出代码,这个代码怎么写算法啊,求教,我真的不会写算法怎么办#incl...

    该楼层疑似违规已被系统折叠 隐藏此楼查看此楼 这个代码怎么写算法啊,求教,我真的不会写算法怎么办 #include "stdio.h" #define N 3 //学生数3. st ...

  6. 指令流水 一个时钟周期 出一个结果_以SM3算法为例,构建一个软硬协作算法加速器:性能分析与优化...

    衡量一款 ASIC 芯片可以从 PPA 三个角度进行. PPA 指的是: Power/Performance/Area,功耗 / 性能 / 面积. 衡量 FPGA 设计同样可以参照 PPA,但又有所不 ...

  7. 若S作主串,P作模式串,试分别写出利用BF算法和KMP算法的匹配过程。

    目   录 题目: 百度文库-答案: (1) (2) MOOC标准答案: (1) (2) mooc答案-截图: 数据结构(C语言版)-严蔚敏2007 题目: 设字符串S='aabaabaabaac', ...

  8. JavaScript实现找出买卖股票的最大利润算法(附完整源码)

    JavaScript实现找出买卖股票的最大利润算法(附完整源码) dpBestTimeToBuySellStocks.js完整源代码 dpBestTimeToBuySellStocks.test.js ...

  9. JavaScript实现找出一个数的质因数primeFactors算法(附完整源码)

    JavaScript实现找出一个数的质因数primeFactors算法(附完整源码) primeFactors.js完整源代码 primeFactors.js完整源代码 export function ...

最新文章

  1. linux mac中实现类似secureCRT的clone session
  2. OpenCV2.4.9 For Android + Android Studio (with gradle)配置教程
  3. 正在被巨大数据中心吞噬的全球电力,谷歌脸书和比特币的“源”罪
  4. ICDM 2020 TOP3方案
  5. 使用pip安装BeautifulSoup4模块
  6. swiper实现局部内容滚动效果
  7. Oracle常用数据字典表
  8. linux 后台运行_Linux系统后台运行应用三板斧
  9. Linux下Ipython安装
  10. 计算机的组成 —— VGA
  11. 如何保护开发人员工作站
  12. VirtualBox文件数量一多,必然崩溃
  13. java连不上mysql_eclipse连接不上mysql
  14. JavaScript,凯撒位移密码,算法
  15. Selenium显示等待和隐式等待
  16. 多系统下的蓝牙设备共用配对问题之LTK、EDIV、ERAND.以 Manjaro、Debian、Windows10 为例
  17. 快手的未来,没有宿华
  18. 在测试集PARSEC测试集简介与使用
  19. [Signal]软阈值与硬阈值
  20. python成绩查询系统_教你用python爬虫监控教务系统,查成绩快人一步!

热门文章

  1. flutter图片聊天泡泡_flutter即时聊天IM仿微信|flutter聊天界面
  2. c语言组成整数的最大数字,c语言 编写程序将一个正整数中的所有偶数数字取出来并用这些数字构成一个最大数。...
  3. 不再为 Node.js 模块自动引入 Polyfills
  4. shouldComponentUpdate 的作用
  5. java session 生命周期_java之hibernate之session中对象的生命周期
  6. java 短链接url_Java 网址短链接服务原理及解决方案
  7. mysql安全方面_MySQL数据库在网络安全方面功能有哪些呢?
  8. django导入mysql_django如何直接对数据库进行插入操作?
  9. 微型计算机没什么总线,微型计算机系统总线包括什么?
  10. 51ctopython自动化测试工程师课程价格,Python自动化测试开发实战 一门能就业的测试课...