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.

7.8 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;
}

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 INF = 0x3f3f3f3f;

const int N = 1003;
int g[N][N];    // 邻接矩阵
int n;          // n个点
int w[N];       // 额外费用
int path[N][N]; // i->j 可能存在多条路线,我要找最短的。如果有多条最短的,我要字典序最小的。现在路线唯一了吧!比如这条路线最终是
// i->a->b->c->d->j,则path[i][j]=a,也就是第一个后继节点。
void floyd() {
    for (int k = 1; k <= n; k++)
        for (int i = 1; i <= n; i++)
            if (g[i][k] != INF) // floyd优化
                for (int j = 1; j <= n; j++) {
                    if (g[i][j] > g[i][k] + g[k][j] + w[k]) { // w[k]:点权
                        g[i][j] = g[i][k] + g[k][j] + w[k];   // k的加入使得i->j的路径变短
                        path[i][j] = path[i][k];              // 如果i->k->j使得i->j更近那么根据定义path[i][j]就是这条最短路径中距离i最近的那个点而这个点由于是出现在i->k的必经之路上而且是i->k的首席弟子所以也必然是i->j的首席弟子。
                    }
                    // 处理字典序
                    if (g[i][j] == g[i][k] + g[k][j] + w[k]) {                // 如果存在多条最短路径也就是除了k还有其它k1,k2使得i->j距离一样小
                        if (path[i][j] > path[i][k]) path[i][j] = path[i][k]; // 字典序,谁更小就留下谁
                    }
                }
}
int main() {
    while (cin >> n && n) {
        for (int i = 1; i <= n; i++) {
            for (int j = 1; j <= n; j++) {
                path[i][j] = j;                   // 路径初始化,记录整条路径上离i节点最近的最短路径上的下一个点只有i->j时下一个点可不就是j
                cin >> g[i][j];                   // 不管是不是有边,都先录进来
                if (g[i][j] == -1) g[i][j] = INF; // 如果题目中给出的是无边那么设置为正无穷。此时有些记录的path[i][j]就是没用的但没事后面会被其它代码替换掉path[i][j]。
            }
        }
        for (int i = 1; i <= n; i++) cin >> w[i]; // 读入点权

        // 多源最短路径
        floyd();

        // 处理询问
        int x, y;
        while (cin >> x >> y) {
            if (x == -1 && y == -1) break;
            printf("From %d to %d :\n", x, y);
            printf("Path: %d", x);
            int u = x, v = y;
            // 理解路径思路:
            // (1) 从起点x出发,用循环打印路径,最后一个打印的肯定是y
            // (2) 从起点x出发,第二个点应该是离x最近的并且是最短路径上的那个点,这个点就是path[x][y]!
            // path[x][y]从起点x出发到终点y有多条最短路径我们选择字典序最小的那条最短路径然后path[x][y]就是从x出发离x最近的这条最短路径上的点。
            while (x != y) {
                printf("-->%d", path[x][y]); // 输出距离x最近的那个点
                x = path[x][y];              // 更换x概念向y逼近让循环跑起来
            }

            puts("");
            if (g[u][v] < INF)
                printf("Total cost : %d\n", g[u][v]);
            else
                puts("-1");
            puts("");
        }
    }
    return 0;
}

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

https://juejin.cn/post/6935691567696969764