You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

11 KiB

This file contains ambiguous Unicode characters!

This file contains ambiguous Unicode characters that may be confused with others in your current locale. If your use case is intentional and legitimate, you can safely ignore this warning. Use the Escape button to highlight these characters.

图论-多源最短路径(Floyd算法)

一、Floyd

Floyd算法是一次性求所有结点之间的最短距离,能处理负权边的图,程序比暴力的DFS更简单,但是复杂度是O(n^3),只适合 n < 200的情况。 Floyd运用了 动态规划 的思想,求 i 、 j两点的最短距离,可分两种情况考虑,即经过图中某个点 k的路径和不经过点 k 的路径,取两者中的最短路径

  • 判断负圈 眼尖的人儿可能发现邻接矩阵 mp 中, mp[i][i]并没有赋初值0,而是 inf。并且计算后 mp[i][i]的值也不是 0,而是 mp[i][i]=mp[i][u]+……+mp[v][i],即从外面绕一圈回来的最短路径,而这正 用于判断负圈,即 mp[i][i]<0

相关变形结合题目讲,如:负圈、打印路径、最小环、传递闭包

记录坑点:重复边,保留最小的那个。

二、模板

void floyd() {
	for (int k = 1; k <= n; k++)
		for (int i = 1; i <= n; i++)
			if (g[i][k] != inf)  //优化
				for (int j = 1; j <= n; j++)
					if (g[i][j] > g[i][k] + g[k][j])
						g[i][j] = g[i][k] + g[k][j];
}

三、判负环

类型 判负环

题意

  • 正常路是m条双向正权边
  • 虫洞是w条单向负权边
  • 题目让判断是否有负权回路

办法 利用Floyd找两点间花费的最短时间,判断从起始位置到起始位置的最短时间是否为负值(判断负权环),若为负值,说明他通过虫洞回到起始位置时比自己最初离开起始位置的时间早。

代码实现: 在第二重循环,求完第i个结点后判断。ii之间的最短距离是一个负值,说明存在一个经过它的负环。

#include <cstdio>
#include <cstring>
#include <algorithm>
#include <iostream>
using namespace std;
const int INF = 0x3f3f3f3f;

const int N = 502;
int n, m, w;
int g[N][N];

// floyd判断是否存在负圈
bool floyd() {
    for (int k = 1; k <= n; k++)
        for (int i = 1; i <= n; i++)
            if (g[i][k] != INF) { // 优化
                for (int j = 1; j <= n; j++)
                    if (g[i][j] > g[i][k] + g[k][j])
                        g[i][j] = g[i][k] + g[k][j];
                if (g[i][i] < 0) return true; // 发现负圈
            }
    return false;
}
int main() {
    int T;
    cin >> T;
    while (T--) {
        cin >> n >> m >> w;
        memset(g, INF, sizeof g); // 初始化邻接矩阵

        // 双向正值边
        while (m--) {
            int a, b, c;
            cin >> a >> b >> c;
            // 注意坑:重边
            g[a][b] = g[b][a] = min(c, g[a][b]);
        }
        // 单向负值边
        while (w--) {
            int a, b, c;
            cin >> a >> b >> c;
            g[a][b] = -c; // 负值边
        }

        if (floyd())
            puts("YES");
        else
            puts("NO");
    }
    return 0;
}

四、记录最短路径并输出

floyd 算法求最短路(边权可为负)很优美,四行代码就搞定了。今天做了一个题,可以用 floyd 做,但是要最短路的路径。在网上搜了一阵,代码倒是有,但是没有解释,为何是这样的?于是,手推了一遍,写了这篇博客。

不像 dijkstraspfa,是一个点一个点加进去的,直接 pre 数组往前倒,倒至起点就行了。floyd 是基于动态规划,这怎么记录路径呢?

开一个 path数组,path[i][j] 表示:更新从 ij 的最短路径时,经过的一个中转点。

void floyd() {
    for (int k = 1; k <= n; k++)
        for (int i = 1; i <= n; i++)
            for (int j = 1; j <= n; j++)
                if (dis[i][j] > dis[i][k] + dis[k][j]) {
                    dis[i][j] = dis[i][k] + dis[k][j];
                    path[i][j] = k; //记录i->j是通过k转移的
                }
}

在这个图中,很容易看出,从 16 之间的最短路径是标红的那几条边。

二重循环所有点,输出 path 数组:

0 0 0 3 4 5
0 0 0 0 4 5
0 0 0 0 4 5
0 0 0 0 0 5
0 0 0 0 0 0
0 0 0 0 0 0

可以看出,path[1][6]5path[1][5]4path[1][4]3。那么,最终 path[i][j] 中存的就是 ij的最短路径中的靠近j的最后一个点

而我们最终输出路径的思路就是,不断分段最短路径! 最后输出所有的点。

原理:由 ij 的最短路径中的一点 k,将最短路径分段为从 ik 的最短路径 和 从 kj 的最短路径,最短路径就为从ik的最短路径+从kj的最短路径,一直分段,直到分到 ij 为同一点,停止

可能现在你有些迷糊,我们直接看代码吧!

void print(int i, int j) { // path[i][j]:从i到j最短路径中经过的一点k
    if (i == j) return;    // 分段到同一点,递归结束
    if (path[i][j] == 0)   // i和j直接相连就是i到j最短路径不经过任何点
        cout << i << " " << j << endl; 
    else {
        print(i, path[i][j]); // 分段输出从i到k的最短路径
        // 输出从i到k最短路径中的所有点一定都在从i到j的最短路径中
        print(path[i][j], j); // 分段输出从k到j的最短路径
        // 输出从j到k最短路径中的所有的点
    }
}

就是一个 递归 的过程。

我们用上面的图模拟一下: 首先,从 16 的最短路径,path[1][6]中存的是:该最短路径中的最后一个节点 5,即,k = 5

那么,递归(分段) 到,从 15的最短路 和 从56的最短路。

  • 15的最短路,path[1][5]=4,则又分段为从14的最短路和从45的最短路。
  • 14的最短路,path[1][4]=3,则又分段为从13的最短路和从34的最短路。
  • path[1][3]=0!如图,13直接相连!那么13都是最短路中的点,输出就行了!

回溯

  • 34的最短路,path[3][4]=034直接相连,34都是最短路中的点,输出!
  • 45的最短路,path[4][5]=045直接相连,45都是最短路中的点,输出!
  • 56的最短路,path[5][6]=056直接相连,56都是最短路中的点,输出!

所以,最终输出的就是:

1 3
3 4
4 5
5 6

依次连接就是从16的最短路径了!

总体代码

void floyd() {
    for (int k = 1; k <= n; k++)
        for (int i = 1; i <= n; i++)
            for (int j = 1; j <= n; j++)
                if (dis[i][j] > dis[i][k] + dis[k][j]) {
                    dis[i][j] = dis[i][k] + dis[k][j];
                    path[i][j] = k;
                }
}
void print(int i, int j) {
    if (i == j) return;
    if (path[i][j] == 0)
        cout << i << " " << j << endl;
    else {
        print(i, path[i][j]);
        print(path[i][j], j);
    }
}

int main() {
    ··· //一顿初始化,输入数据

    floyd();
    print(1, n); // 输出从1到n的最短路径中的所有点

    return 0;
}

练习题

HDU-1385 Minimum Transport Cost

类型 打印路径

题意 给你所有城市到其他城市的道路成本和经过每个城市的城市税,给你很多组城市,要求你找出每组城市间的最低运输成本并且输出路径,如果有多条路径则输出字典序最小的那条路径注意,起点城市和终点城市不需要收城市税(中间点才收税,也就是插值的k收税)。

分析 输出路径,多个答案则输出字典序最小的,无法到达输出-1。 读入邻接表, w[]记录每个城市额外费用, path[][]记录路径,floyd()里维护即可。然后处理下输出(比较恶心)。

解释int path[N][N]; i \rightarrow j 可能存在多条路线,我要找最短的。如果有多条最短的,我要字典序最小的。现在路线唯一了吧!比如这条路线最终是 i \rightarrow a \rightarrow b \rightarrow c \rightarrow d \rightarrow j,则path[i][j]=a,也就是第一个后继节点。

#include <bits/stdc++.h>
using namespace std;

const int N = 110;
const int INF = 0x3f3f3f3f;
// Floyd+记录起点后继
int n;
int g[N][N], w[N];
int path[N][N]; // 记录i到j最短路径中i的后继

void floyd() {
    for (int k = 1; k <= n; k++)
        for (int i = 1; i <= n; i++)
            for (int j = 1; j <= n; j++) {
                if (g[i][j] > g[i][k] + g[k][j] + w[k]) {
                    g[i][j] = g[i][k] + g[k][j] + w[k];
                    path[i][j] = path[i][k]; // i->j这条最短路径上i后面第一个节点是i->k路径上第一个节点
                }
                // 相同路径下选择后继更小的(为了字典序)
                if (g[i][j] == g[i][k] + g[k][j] + w[k])
                    if (path[i][j] > path[i][k])
                        path[i][j] = path[i][k];
            }
}

// 递归输出路径
void print(int s, int e) {
    printf("-->%d", path[s][e]); // 输出s的后继
    if (path[s][e] != e)         // 如果不是直连
        print(path[s][e], e);    // 递归输出后继
}

int main() {
    while (cin >> n, n) {
        for (int i = 1; i <= n; i++)
            for (int j = 1; j <= n; j++) {
                cin >> g[i][j];
                if (g[i][j] == -1) g[i][j] = INF;
                path[i][j] = j;
            }

        for (int i = 1; i <= n; i++) cin >> w[i];
        floyd();

        int s, e;

        while (cin >> s >> e, ~s && ~e) {
            printf("From %d to %d :\n", s, e);
            printf("Path: %d", s);
            if (s != e) print(s, e); // 起点与终点不同开始递归
            printf("\nTotal cost : %d\n\n", g[s][e]);
        }
    }
    return 0;
}

TODO 据说可以使用Dijkstra算法解决,有空可以试试: 链接

https://juejin.cn/post/6935691567696969764