全源最短路问题:Floyd算法详解【经典算法,限时免费】

news2025/1/4 18:33:51

文章目录

  • 最短路问题概述
  • 带边权的图的全源最短路径
  • Floyd算法解决全源最短路问题
    • `dist`数组初始化
    • `dist`数组迭代以及动态转移方程
    • Floyd算法求解dist数组完整代码
      • python
      • java
      • cpp
      • 时空复杂度
  • *Floyd算法正确性证明
    • 证明过程
      • 初始情况
      • 归纳假设
      • 归纳步骤
      • 终止条件
    • 完整证明
  • 相关题目

最短路问题概述

最短路问题(Shortest Path Problem)是图论中一个经典的问题,旨在找到从一个顶点到另一个顶点的最短路径。

所给定的图可以是有向图(directed graph)也可以是无向图(undirected graph),并且边可以有权重(weights),即每条边有一个数值表示从一个顶点到另一个顶点的距离或成本。

最短路问题的常见变种包括:

  1. 单源最短路径问题:找到从图中某个特定顶点到所有其他顶点的最短路径。
  2. 单对最短路径问题:找到从图中某个特定顶点到另一个特定顶点的最短路径。
  3. 全源最短路径问题:找到图中每对顶点之间的最短路径。

解决最短路问题通常有包含以下算法

  1. Dijkstra算法
  • 定义:用于解决单源最短路径问题,适用于边权重非负的图。
  • 原理:通过贪心策略,逐步选择具有最小估计距离的顶点,并更新其邻接顶点的距离。
  • 时间复杂度:O((V + E) log V),其中 V 是顶点数,E 是边数。
  1. Bellman-Ford算法
  • 定义:用于解决单源最短路径问题,可以处理负权重的边。
  • 原理:通过对所有边进行多次松弛操作,逐步逼近最短路径。
  • 时间复杂度:O(VE),其中 V 是顶点数,E 是边数。
  1. Floyd算法
  • 定义:用于解决全源最短路径问题
  • 原理:通过动态规划的方式,逐步考虑每个顶点作为中间点,更新所有顶点对之间的最短路径。
  • 时间复杂度:O(V^3),其中 V 是顶点数。

本篇讲解主要介绍Floyd算法的原理以及应用

带边权的图的全源最短路径

对于以下无向图,我们能够看到图中包含了若干节点以及节点之间的边权

这里的边权可以简单理解为两个节点之间的距离

在这里插入图片描述

大部分题目(但不是绝对的)会用一个外层长度为n,内层长度为3的二维数组edges,来表示包含图。

其中n是整个图的边数,edges[i] = [start_i, end_i, w_i],表示在第i条边中,start_i节点和end_i节点之间的距离为w_i

譬如,上述无向图可以用以下二维数组edges来表示

edges = 
[
[0, 1, 2],
[0, 4, 8],
[1, 2, 3],
[1, 4, 2],
[2, 3, 1],
[3, 4, 1]
]

有几点需要注意:

  1. n = len(edges) = 6 是该无向图的边数
  2. edges[i]edges中的顺序并不重要
  3. 对于无向图而言,每一条边的起点start_i和终点end_i的顺序也不重要
  4. 如果是一个有向图,可能存在一条边的起点和终点互换则边权值不相等的情况出现(在实际生活中表现为两点之间的路径存在单行道、红灯停留时间等等)

假设我们现在需要一个储存数据的结构,通过该结构能够快速查询图中任意两个点的最短路径,那么这个数据应该如何设计?

由于查询任意两个点之间的距离都必须知道这两个点的具体编号,容易想到可以设计如下的一个二维数组。

在这里插入图片描述

我们可以把这个二维数组称为dist,其中行索引为起点的编号(绿色),列索引为终点的编号(黄色)。

distdistance的缩写

如果我们要查询节点i到节点j的最短路径,我们直接通过查找dist[i][j]的值就可以得到。

譬如,dist[0][3]表示从节点0到节点3的最短路径为5

在图中,我们容易看出这条路径是0 -> 1 -> 4 -> 3

在这里插入图片描述

容易注意到二维数组dist存在以下特点

  1. 该二维数组的内层和外层均为nnn列),表示该数组包含了n*n对节点的最短路径查询情况。
  2. 该二维数组主对角线的值均为0(灰色部分),即存在dist[i][i] = 0成立,这表示任意一个节点到达它自身的最短路径是0(不需要任何其他路径)
  3. 该二维数组沿主对角线对称,即存在dist[i][j] = dist[j][i]成立,表示从节点i出发到节点j的最短路径,和从节点j出发到节点i的最短路径是相同的。(这个特点仅对无向图成立)换句话说,只需要看这个二维数组的右上部分(紫色)或左下部分(白色),即可得知所有信息。

Floyd算法解决全源最短路问题

截止到目前,我们已经知道,解决全源最短路问题,要求我们得到的结果就是这个dist

接下来努力的方向就是如何找到这个dist

而Floyd算法就是基于动态规划思想,通过多次迭代找到这个二维数组dist的过程。

dist数组初始化

回到例子

edges = [
[0, 1, 2],
[0, 4, 8],
[1, 2, 3],
[1, 4, 2],
[2, 3, 1],
[3, 4, 1]
]

我们根据上述edges数组所给出的起始边情况,初始化dist数组。若

  • 节点start_i和节点end_iedges中是一条初始存在的边,则设置dist[start_i][end_i]为其对应的权值或距离w_i
  • 节点start_i和节点end_iedges中不是一条初始存在的边,则设置dist[start_i][end_i]为正无穷inf或一个极大的数,表示在初始边中,无法从start_i到达end_i
  • 节点start_iend_i相等的时候,则设置dist[start_i][end_i] = 0,即对角线上的0

暂时无法在飞书文档外展示此内容

其对应代码如下

from math import inf

def Floyd(n, edges):
    
    dist = [[inf] * n for _ in range(n)]
    
    for start, end, w in edges:
        dist[start][end], dist[end][start] = w, w
    
    for i in range(n):
        dist[i][i] = 0
    
    pass

dist数组迭代以及动态转移方程

由于初始化后dist数组中的数字仅仅代表着这些直接连接的边的长度,我们需要思考,如果我们把某些点作为桥梁(或称之为中间点)把某两个点连起来,那么会不会使得这两个点之间的最短路径相较于之前变得更短?

换句话说,假设我们选择节点k,作为节点i和节点j之间所经过的中间点,能否使得节点i和节点j之间的最短路径dist[i][j]变得更小?

譬如,考虑节点1作为节点0和节点4之间的中间点。

由于dist[0][1] = 2, dist[1][4] = 2dist[0][4] = 8

我们发现存在dist[0][1] + dist[1][4] < dist[0][4]成立。

这意味着,如果我们想从节点0出发到达节点4,走0 -> 1 -> 4这样的路径,会优于0 -> 4这样的路径。

暂时无法在飞书文档外展示此内容

所以,我们会将dist[0][4]修改为比之前更小的dist[0][1] + dist[1][4] = 4

暂时无法在飞书文档外展示此内容

而在这之后,当我们涉及到以节点4作为中间节点去连接从节点0出发到其他节点(譬如节点3)的时候,就可以使用dist[0][4] = 4而不是dist[0][4] = 8来作为节点0到节点4的最短路径了。

随着这个例子的推出,Floyd算法的动态转移方程也就呼之欲出了。

假设我们选择节点k,作为节点i和节点j之间所经过的中间点。那么

d i s t [ i ] [ j ] = m i n ⁡ ( d i s t [ i ] [ j ] , d i s t [ i ] [ k ] + d i s t [ k ] [ j ] ) dist[i][j]=min⁡(dist[i][j],dist[i][k]+dist[k][j]) dist[i][j]=min(dist[i][j],dist[i][k]+dist[k][j])

换句话说,当考虑经过中间点k能使得节点i和节点j之间的路径更短,则修改dist[i][j]为这个更短距离。

由于每一个节点都有可能成为另外任意两个节点的中间节点,为了不漏掉任何一个可能的中间节点,我们必须对所有的中间节点k进行遍历,并且在固定k的前提下,再次进行起点i和终点j的双重遍历。

其对应的代码为

def Floyd(n, edges):
    
    pass
    
    for k in range(n):
        for i in range(n):
            for j in range(n):
                dist[i][j] = min(dist[i][j], dist[i][k] + dist[k][j])
    
    return dist

Floyd算法求解dist数组完整代码

python

def Floyd(n, edges):
    
    # 初始化dist二维数组,用一个大数代替inf
    dist = [[100000] * n for _ in range(n)]
    
    # 初始化dist二维数组中的边为edges的起始边的情况
    for start, end, w in edges:
        dist[start][end], dist[end][start] = w, w
    
    # 初始化dist二维数组的对角线均为0
    for i in range(n):
        dist[i][i] = 0
    
    # 遍历中间节点k,每一个节点都有可能成为中间节点
    for k in range(n):
        # 遍历起始节点i,每一个节点都有可能成为起始节点
        for i in range(n):
            # 遍历终止节点j,每一个节点都有可能成为终止节点
            for j in range(n):
                # 动态转移方程
                dist[i][j] = min(dist[i][j], dist[i][k] + dist[k][j])

    # 返回dist数组    
    return dist

java

import java.util.Arrays;

public class Main {
    public static int[][] Floyd(int n, int[][] edges) {
        // 初始化dist二维数组,用一个大数代替inf
        int[][] dist = new int[n][n];
        for (int[] row : dist) {
            Arrays.fill(row, 100000);
        }

        // 初始化dist二维数组中的边为edges的起始边的情况
        for (int[] edge : edges) {
            int start = edge[0];
            int end = edge[1];
            int w = edge[2];
            dist[start][end] = w;
            dist[end][start] = w;
        }

        // 初始化dist二维数组的对角线均为0
        for (int i = 0; i < n; i++) {
            dist[i][i] = 0;
        }

        // 遍历中间节点k,每一个节点都有可能成为中间节点
        for (int k = 0; k < n; k++) {
            // 遍历起始节点i,每一个节点都有可能成为起始节点
            for (int i = 0; i < n; i++) {
                // 遍历终止节点j,每一个节点都有可能成为终止节点
                for (int j = 0; j < n; j++) {
                    // 动态转移方程
                    dist[i][j] = Math.min(dist[i][j], dist[i][k] + dist[k][j]);
                }
            }
        }

        // 返回dist数组
        return dist;
    }

    public static void main(String[] args) {
        // 初始化n和edges数组,可以自行进行修改和调整
        int n = 5;

        int[][] edges = {
            {0, 1, 2},
            {0, 4, 8},
            {1, 2, 3},
            {1, 4, 2},
            {2, 3, 1},
            {3, 4, 1}
        };

        // 调用Floyd函数得到dist最小距离矩阵
        int[][] dist = Floyd(n, edges);

        // 输出dist最小距离矩阵
        for (int[] row : dist) {
            for (int val : row) {
                System.out.print(val + " ");
            }
            System.out.println();
        }
    }
}

cpp

#include <iostream>
#include <vector>
#include <algorithm>

using namespace std;

vector<vector<int>> Floyd(int n, vector<vector<int>>& edges) {
    // 初始化dist二维数组,用一个大数代替inf
    vector<vector<int>> dist(n, vector<int>(n, 100000));

    // 初始化dist二维数组中的边为edges的起始边的情况
    for (const auto& edge : edges) {
        int start = edge[0];
        int end = edge[1];
        int w = edge[2];
        dist[start][end] = w;
        dist[end][start] = w;
    }

    // 初始化dist二维数组的对角线均为0
    for (int i = 0; i < n; i++) {
        dist[i][i] = 0;
    }

    // 遍历中间节点k,每一个节点都有可能成为中间节点
    for (int k = 0; k < n; k++) {
        // 遍历起始节点i,每一个节点都有可能成为起始节点
        for (int i = 0; i < n; i++) {
            // 遍历终止节点j,每一个节点都有可能成为终止节点
            for (int j = 0; j < n; j++) {
                // 动态转移方程
                dist[i][j] = min(dist[i][j], dist[i][k] + dist[k][j]);
            }
        }
    }

    // 返回dist数组
    return dist;
}

int main() {
    // 初始化n和edges数组,可以自行进行修改和调整
    int n = 5;

    vector<vector<int>> edges = {
        {0, 1, 2},
        {0, 4, 8},
        {1, 2, 3},
        {1, 4, 2},
        {2, 3, 1},
        {3, 4, 1}
    };

    // 调用Floyd函数得到dist最小距离矩阵
    vector<vector<int>> dist = Floyd(n, edges);

    // 输出dist最小距离矩阵
    for (const auto& row : dist) {
        for (int val : row) {
            cout << val << " ";
        }
        cout << endl;
    }

    return 0;
}

时空复杂度

时间复杂度:O(N^3)。三重循环所需时间复杂度。

空间复杂度:O(N^2)。最小距离矩阵dist所占空间。

其中N为节点数目。

*Floyd算法正确性证明

本处内容不做具体讲解,感兴趣的同学可以自行学习。

Floyd算法的核心是其动态转移方程

d i s t [ i ] [ j ] = m i n ⁡ ( d i s t [ i ] [ j ] , d i s t [ i ] [ k ] + d i s t [ k ] [ j ] ) dist[i][j]=min⁡(dist[i][j],dist[i][k]+dist[k][j]) dist[i][j]=min(dist[i][j],dist[i][k]+dist[k][j])

该转移方程表示,如果通过顶点 k 作为中间点可以使 ij 的路径更短,则更新 dist[i][j]dist[i][k] + dist[k][j]

证明过程

我们可以使用归纳法来证明Floyd算法的正确性。

初始情况

在初始情况下, dist[i][j] 被设定为以下几种情况之一:

  1. 如果 i == j,则 dist[i][j] = 0
  2. 如果 ij 之间有一条边,则 dist[i][j] 为该边的权重。
  3. 否则, dist[i][j] 设为无穷大(inf),表示两点之间不可达。

这个初始化保证了在没有中间点的情况下,每对顶点 (i, j) 的最短路径是正确的。

归纳假设

假设在迭代到某个中间点 k 之前,dist[i][j] 表示从顶点 i 到顶点 j 的最短路径,且这个路径仅包含编号不大于 k-1 的中间点。

归纳步骤

对于每一对顶点 (i, j),我们在考虑是否通过中间点 k 更新路径时有以下两种情况:

  1. 不通过中间点 k:在这种情况下,最短路径长度保持为 dist[i][j],即没有比原有路径更短的路径。
  2. 通过中间点 k:如果通过 k 能够使路径更短,则路径长度应更新为 dist[i][k] + dist[k][j]。这是因为 dist[i][k] 是从 ik 的最短路径长度,而 dist[k][j] 是从 kj 的最短路径长度。

根据动态转移方程:

d i s t [ i ] [ j ] = m i n ⁡ ( d i s t [ i ] [ j ] , d i s t [ i ] [ k ] + d i s t [ k ] [ j ] ) dist[i][j]=min⁡(dist[i][j],dist[i][k]+dist[k][j]) dist[i][j]=min(dist[i][j],dist[i][k]+dist[k][j])

我们选择较小的值来更新 dist[i][j],确保 dist[i][j] 始终表示从 ij 的最短路径长度。

终止条件

当所有顶点都作为中间点被考虑完毕后,dist[i][j] 就表示从 ij 的最短路径长度,因为所有可能的中间点都已经被考虑在内。

完整证明

  1. 初始条件正确:初始化阶段保证了直接路径的正确性。
  2. 保持最优子结构性质:通过动态转移方程,我们在每一步中都选择最优解(较小的路径长度),确保子问题的解是最优的。
  3. 正确性归纳:在迭代的每一步中,假设之前的所有步骤已经正确计算了路径长度,那么当前步骤也能正确更新路径长度,从而保证最终结果的正确性。

通过上述步骤,我们证明了Floyd算法的动态转移方程的正确性。该方程在每一步中都能够有效地更新最短路径长度,最终保证在所有中间点被考虑之后,dist[i][j] 表示从顶点 i 到顶点 j 的最短路径长度。

相关题目

LeetCode1334.阈值距离内邻居最少的城市

【最短路问题】2024D-快递员的烦恼

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1969546.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

log4j2 远程代码执行漏洞复现(CVE-2021-44228)

目录 一、相关概念 二、漏洞复现 步骤一&#xff1a;执行一下命令启动靶场环境并在浏览器访问&#xff01;&#xff01;&#xff01; 1.启动命令 2.访问网址 步骤二&#xff1a;先在自己搭建的DNSLOG平台上获取一个域名来监视我们注入的效果... 步骤三:可以发现 /solr/ad…

小米手机通知栏横条无法清除 刻晴主题 历史通知记录功能使用方案

问题环境&#x1f495; Root K50 Ultra HyperCeiler &#xff08;系统修改&#xff09; Lspose&#xff08;框架&#xff09;魔改刻晴主题&#xff08;自用&#xff09;隐藏&#xff08;最近任务&#xff09;后台 在搞机过程中出现过如下情况 通知条无法被清除&#xff0c;导…

本地索引和全局索引视图测试

1、创建分区表 CREATE TABLE "OSGIC"."TT" ( "ID" VARCHAR2(32) NOT NULL ENABLE, "BUSS_ID" VARCHAR2(64), "REEST_TIME" DATE ) PARTITION BY RANGE ("REEST_TIME") INTERVAL(NUMTOYMINTERVAL(1, MONTH)) (…

Redis部署策略

文章目录 Redis部署策略单机部署主从复制哨兵模式Redis集群 Redis部署策略 部署方式描述单机部署适用于对数据可靠性要求不高、规模较小的应用。部署简单&#xff0c;但没有冗余和高可用性。主从复制数据从主节点同步到一个或多个从节点&#xff0c;提升读性能和数据备份能力。…

堆的实现-向上调整算法-向下调整算法-堆排序-TopK问题 C语言

一、堆的概念及结构 二、 向上调整算法 注意:循环条件不可写parent > 0 //向上调整算法 //child为下标 void adjustup(int* a, int child) {int parent (child - 1) / 2;while (child > 0){if (a[child] < a[parent]){swap(&a[child], &a[parent]);child pa…

免费开源的私人笔记项目,极空间搭建全平台笔记和待办事项『Joplin』

免费开源的私人笔记项目&#xff0c;极空间搭建全平台笔记和待办事项『Joplin』 哈喽小伙伴们好&#xff0c;我是Stark-C~ 我想很多小伙伴都有记录笔记或者备忘的习惯&#xff0c;其实目前市面可用的这类应用还是很多的&#xff0c;比如印象笔记&#xff0c;有道云笔记。作为…

文件解析漏洞集合

IIS解析漏洞 IIS6 目录解析 在网站下建立文件夹的名字为.asp/.asa 的文件夹&#xff0c;其目录内的任何扩展名的文件都被IIS当作asp 文件来解析并执行。 这里显示的是 1.asp下的1.jpg&#xff0c;按照道理来说里面的文件是一个图片&#xff0c;但是访问的话&#xff0c;会出…

SpringBoot面试题整理(1)

面试整理 前置知识 ApplicationContextInitializerApplicationListenerBeanFactoryBeanDefinitionBeanFactoryPostProcessorAwareInitializingBean/DisposableBeanBeanPostProcessor 面试题 SpringBoot启动流程IOC容器初始化流程Bean声明周期Bean循环依赖SpringMVC执行流程…

JavaEE 初阶(13)——多线程11之“定时器”

目录 一. 什么是“定时器” 二. 标准库的定时器 三. 定时器的实现 MyTimer 3.1 分析思路 1. 创建执行任务的类。 2. 管理任务 3. 执行任务 3.2 线程安全问题 四. 拓展 一. 什么是“定时器” 定时器是软件开发中的一个重要组件&#xff0c;类似于一个“闹钟”&#xff0…

【Spring】详细理解Spring中控制反转(IOC)和依赖注入(DI)的设计思想。

目录 1.相关概念 2. 设计IOC容器的两个主要接口 2.1 BeanFactory接口 2.2 ApplicationContext接口 2.3 两个接口的区别【面试题】 3. Bean对象的管理&#xff08;控制反转&#xff09; 3.1 基于XML文件管理bean对象 3.2 基于注解方式去管理【重点】 4. 依赖注入&#…

81.WEB渗透测试-信息收集-框架组件识别利用(5)

免责声明&#xff1a;内容仅供学习参考&#xff0c;请合法利用知识&#xff0c;禁止进行违法犯罪活动&#xff01; 内容参考于&#xff1a; 易锦网校会员专享课 上一个内容&#xff1a;80.WEB渗透测试-信息收集-框架组件识别利用&#xff08;4&#xff09; solr&#xff1a;…

第三方库jsoncpp

文章目录 0.jsoncpp库是做什么的&#xff1f;1.安装库2.有哪几个类&#xff0c;如何使用Json::Value类Json::Writer类-StreamWriterJson::Reader类-CharReader三者关系 3.使用样例将数据先存入Value类&#xff0c;再通过StreamWriter类转为Json格式的字符串获取到Json格式字符串…

6 postgresql事务与并发控制

事务与并发控制 事务可以理解为做一件事&#xff0c;数据库就是一件事要么做完&#xff0c;要么不做&#xff0c;不然这数据库给给人不可靠的感觉 正如解释事务具有四个重要特性&#xff1a;ACID&#xff08;原子性&#xff0c;一致性&#xff0c;隔离性&#xff0c;持久性&a…

python实现提取视频帧的图片

文章目录 1、需求痛点2、完整代码⭐3、代码分析3.1、需要改动的地方3.2、OpenCV库的使用3.3、多线程技术 4、执行效率5、效果展示⭐6、注意事项&#x1f53a;7、总结 &#x1f343;作者介绍&#xff1a;双非本科大三网络工程专业在读&#xff0c;阿里云专家博主&#xff0c;专注…

3 IIC总线

3 IIC总线 1、基本概念1.1 IIC总线定义1.2 IIC总线协议概念 2 以AT24C02为例说明时序2.1 基本特性2.2 利用GPIO模拟IIC2.3 对AT24C02的操作2.4 重定向printf 1、基本概念 1.1 IIC总线定义 定义&#xff1a;两线式串行总线 两线式&#xff1a;说明处理器和外设之间只需两根信号…

【时间复杂度和空间复杂度】(内含超多实例练习)

【时间复杂度和空间复杂度】&#xff08;内含超多实例练习&#xff09; 1. 算法效率2. 时间复杂度2.1 时间复杂度的概念2.2 实例练习2.2.1 数组中搜索数据2.2.2 冒泡排序2.2.3 二分查找2.2.4 阶乘递归2.2.5 斐波那契递归 3. 空间复杂度3.1 空间复杂度的概念3.2 实例练习3.2.1 冒…

Set Hashset底层原理 LinkedSet底层原理 Treeset 31

不重复只执行一次&#xff0c;排序是小到大&#xff0c;色图是接口类 Hashset底层原理 LinkedSet底层原理 Treeset 总结

AI+云边端协同,EasyCVR视频汇聚技术赋能安防监控新生态

随着信息技术的飞速发展和数字化时代的到来&#xff0c;安防监控领域的技术也在不断创新和突破。EasyCVR平台的视频汇聚技术作为其中的佼佼者&#xff0c;以其强大的视频处理、汇聚与融合能力&#xff0c;在安防监控领域展现出了巨大的应用潜力和价值。本文将详细介绍EasyCVR视…

权限束缚术:windows工具自动化权限提升

前言 欢迎来到我的博客 个人主页:北岭敲键盘的荒漠猫-CSDN博客 本文整理利用工具全自动化提权的相关操作方法 因为我没有相关的环境复现&#xff0c;所以没有成功 不过复现的操作就是下面整理的操作 溢出漏洞提权原理 利用溢出漏洞提权&#xff0c;实际上就是一次判断过往的…

极简聊天室-websocket版

再写一个极简聊天室的websocket版&#xff0c;在本例中&#xff0c;websocket仅用于服务器向客户端传输信息&#xff0c;客户端向服务器发送信息是传统的http post方式&#xff0c;用axios来实现的&#xff0c;当然websocket本身是支持双向通信&#xff0c;主要是为了方便跟前面…