• FloydWarshall算法详解(转)


    Floyd-Warshall算法,简称Floyd算法,用于求解任意两点间的最短距离,时间复杂度为O(n^3)。我们平时所见的Floyd算法的一般形式如下:

    1 void Floyd(){
    2     int i,j,k;
    3     for(k=1;k<=n;k++)
    4         for(i=1;i<=n;i++)
    5             for(j=1;j<=n;j++)
    6                 if(dist[i][k]+dist[k][j]<dist[i][j])
    7                     dist[i][j]=dist[i][k]+dist[k][j];
    8 }

      注意下第6行这个地方,如果dist[i][k]或者dist[k][j]不存在,程序中用一个很大的数代替。最好写成if(dist[i][k]!=INF && dist[k][j]!=INF && dist[i][k]+dist[k][j]<dist[i][j]),从而防止溢出所造成的错误。
      上面这个形式的算法其实是Floyd算法的精简版,而真正的Floyd算法是一种基于DP(Dynamic Programming)的最短路径算法。
      设图G中n 个顶点的编号为1到n。令c [i, j, k]表示从i 到j 的最短路径的长度,其中k 表示该路径中的最大顶点,也就是说c[i,j,k]这条最短路径所通过的中间顶点最大不超过k。因此,如果G中包含边<i, j>,则c[i, j, 0] =边<i, j> 的长度;若i= j ,则c[i,j,0]=0;如果G中不包含边<i, j>,则c (i, j, 0)= +∞。c[i, j, n] 则是从i 到j 的最短路径的长度。
      对于任意的k>0,通过分析可以得到:中间顶点不超过k 的i 到j 的最短路径有两种可能:该路径含或不含中间顶点k。若不含,则该路径长度应为c[i, j, k-1],否则长度为 c[i, k, k-1] +c [k, j, k-1]。c[i, j, k]可取两者中的最小值。
      状态转移方程:c[i, j, k]=min{c[i, j, k-1], c [i, k, k-1]+c [k, j, k-1]},k>0。
      这样,问题便具有了最优子结构性质,可以用动态规划方法来求解。

      为了进一步理解,观察上面这个有向图:若k=0, 1, 2, 3,则c[1,3,k]= +∞;c[1,3,4]= 28;若k = 5, 6, 7,则c [1,3,k] = 10;若k=8, 9, 10,则c[1,3,k] = 9。因此1到3的最短路径长度为9。
      下面通过程序来分析这一DP过程,对应上面给出的有向图:

     

     1 #include <iostream>
     2 using namespace std;
     3 
     4 const int INF = 100000;
     5 int n=10,map[11][11],dist[11][11][11];
     6 void init(){
     7     int i,j;
     8     for(i=1;i<=n;i++)
     9         for(j=1;j<=n;j++)
    10             map[i][j]=(i==j)?0:INF;
    11     map[1][2]=2,map[1][4]=20,map[2][5]=1;
    12     map[3][1]=3,map[4][3]=8,map[4][6]=6;
    13     map[4][7]=4,map[5][3]=7,map[5][8]=3;
    14     map[6][3]=1,map[7][8]=1,map[8][6]=2;
    15     map[8][10]=2,map[9][7]=2,map[10][9]=1;
    16 }
    17 void floyd_dp(){
    18     int i,j,k;
    19     for(i=1;i<=n;i++)
    20         for(j=1;j<=n;j++)
    21             dist[i][j][0]=map[i][j];
    22     for(k=1;k<=n;k++)
    23         for(i=1;i<=n;i++)
    24             for(j=1;j<=n;j++){
    25                 dist[i][j][k]=dist[i][j][k-1];
    26                 if(dist[i][k][k-1]+dist[k][j][k-1]<dist[i][j][k])
    27                     dist[i][j][k]=dist[i][k][k-1]+dist[k][j][k-1];
    28             }
    29 }
    30 int main(){
    31     int k,u,v;
    32     init();
    33     floyd_dp();
    34     while(cin>>u>>v,u||v){
    35         for(k=0;k<=n;k++){
    36             if(dist[u][v][k]==INF) cout<<"+∞"<<endl;
    37             else cout<<dist[u][v][k]<<endl;
    38         }
    39     }
    40     return 0;
    41 }

      输入 1 3
      输出 +∞
                +∞
                +∞
                +∞
                28
                10
                10
                10
                9
                9
                9

      Floyd-Warshall算法不仅能求出任意2点间的最短路径,还可以保存最短路径上经过的节点。下面用精简版的Floyd算法实现这一过程,程序中的图依然对应上面的有向图。

     1 #include <iostream>
     2 using namespace std;
     3 
     4 const int INF = 100000;
     5 int n=10,path[11][11],dist[11][11],map[11][11];
     6 void init(){
     7     int i,j;
     8     for(i=1;i<=n;i++)
     9         for(j=1;j<=n;j++)
    10             map[i][j]=(i==j)?0:INF;
    11     map[1][2]=2,map[1][4]=20,map[2][5]=1;
    12     map[3][1]=3,map[4][3]=8,map[4][6]=6;
    13     map[4][7]=4,map[5][3]=7,map[5][8]=3;
    14     map[6][3]=1,map[7][8]=1,map[8][6]=2;
    15     map[8][10]=2,map[9][7]=2,map[10][9]=1;
    16 }
    17 void floyd(){
    18     int i,j,k;
    19     for(i=1;i<=n;i++)
    20         for(j=1;j<=n;j++)
    21             dist[i][j]=map[i][j],path[i][j]=0;
    22     for(k=1;k<=n;k++)
    23         for(i=1;i<=n;i++)
    24             for(j=1;j<=n;j++)
    25                 if(dist[i][k]+dist[k][j]<dist[i][j])
    26                     dist[i][j]=dist[i][k]+dist[k][j],path[i][j]=k;
    27 }
    28 void output(int i,int j){
    29     if(i==j) return;
    30     if(path[i][j]==0) cout<<j<<' ';
    31     else{
    32         output(i,path[i][j]);
    33         output(path[i][j],j);
    34     }
    35 }
    36 int main(){
    37     int u,v;
    38     init();
    39     floyd();
    40     while(cin>>u>>v,u||v){
    41         if(dist[u][v]==INF) cout<<"No path"<<endl;
    42         else{
    43             cout<<u<<' ';
    44             output(u,v);
    45             cout<<endl;
    46         }
    47     }
    48     return 0;
    49 }

      输入 1 3                    
      输出 1 2 5 8 6 3

    作者:BuildNewApp
    出处:http://syxchina.cnblogs.comBuildNewApp.com
    本文版权归作者、博客园和百度空间共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则作者会诅咒你的。
    如果您阅读了我的文章并觉得有价值请点击此处,谢谢您的肯定1。
  • 相关阅读:
    批处理实现终端安全设置
    【干货】JavaScript DOM编程艺术学习笔记10-12【完】
    【干货】JavaScript DOM编程艺术学习笔记7-9
    【干货】JavaScript DOM编程艺术学习笔记4-6
    【干货】JavaScript DOM编程艺术学习笔记1-3
    【干货】Html与CSS入门学习笔记12-14【完】
    Less-8
    Less 7
    Less 6
    Less 5
  • 原文地址:https://www.cnblogs.com/syxchina/p/2197357.html
Copyright © 2020-2023  润新知