344 观光之旅(floyd算法求解最小环)

1. 问题描述:

给定一张无向图,求图中一个至少包含 3 个点的环,环上的节点不重复,并且环上的边的长度之和最小。该问题称为无向图的最小环问题。你需要输出最小环的方案,若最小环不唯一,输出任意一个均可。

输入格式

第一行包含两个整数 N 和 M,表示无向图有 N 个点,M 条边。接下来 M 行,每行包含三个整数 u,v,l,表示点 u 和点 v 之间有一条边,边长为 l。

输出格式

输出占一行,包含最小环的所有节点(按顺序输出),如果不存在则输出 No solution.。

数据范围

1 ≤ N ≤ 100,
1 ≤ M ≤ 10000,
1 ≤ l < 500

输入样例:

5 7
1 4 1
1 3 300
3 1 10
1 2 16
2 3 100
2 5 15
5 3 20

输出样例:

1 3 5 2
来源:https://www.acwing.com/problem/content/description/346/

2. 思路分析:

3. 代码如下:

from typing import List


class Solution:
    # 使用一个全局变量来记录当前环中的对应位置
    count = 0

    def getPath(self, i: int, j: int, path: List[int], pos: List[List[int]]):
        if pos[i][j] == 0: return
        k = pos[i][j]
        # 类似于中序遍历的顺序可以使得得到的环的编号是按顺序的
        self.getPath(i, k, path, pos)
        path[self.count] = k
        self.count += 1
        self.getPath(k, j, path, pos)

    def floyd(self, n: int, g: List[List[int]], dis: List[List[int]]):
        res = 10 ** 10
        path = [0] * n
        # pos记录两点之间的最短距离是由哪一个点转移得到的
        pos = [[0] * (n + 1) for i in range(n + 1)]
        for k in range(1, n + 1):
            # 在floyd算法求解的过程中当前其实得到的是1~k-1编号的任意两点之间的最短路径
            for i in range(1, k):
                for j in range(i + 1, k):
                    if dis[i][j] + g[j][k] + g[k][i] < res:
                        # 注意环是按顺序的, 所以需要特别注意下面的代码的编号顺序
                        res = dis[i][j] + g[i][k] + g[k][j]
                        self.count = 0
                        self.getPath(i, j, path, pos)
                        path[self.count] = j
                        self.count += 1
                        path[self.count] = k
                        self.count += 1
                        path[self.count] = i
                        self.count += 1

            for i in range(1, n + 1):
                for j in range(1, n + 1):
                    if dis[i][j] > dis[i][k] + dis[k][j]:
                        dis[i][j] = dis[i][k] + dis[k][j]
                        # pos记录的k是最小的k
                        pos[i][j] = k
        INF = 10 ** 10
        if res == INF: print("No solution.")
        else:
            for i in range(self.count): print(path[i], end=" ")

    def process(self):
        n, m = map(int, input().split())
        INF = 10 ** 10
        # 使用邻接表存储会比较方便一点
        g = [[INF] * (n + 1) for i in range(n + 1)]
        for i in range(1, n + 1):
            # 自己到自己的点距离为0
            g[i][i] = 0
        for i in range(m):
            x, y, z = map(int, input().split())
            g[x][y] = g[y][x] = min(g[x][y], z)
        # 拷贝二维列表到dis中
        dis = [x[:] for x in g]
        self.floyd(n, g, dis)


if __name__ == "__main__":
    Solution().process()
上一篇:344. 反转字符串(简单)


下一篇:树上倍增法,LCA——CF-832-D