0

0

c++怎么实现弗洛伊德算法求最短路径_c++ 动态规划与邻接矩阵更新【案例】

冰火之心

冰火之心

发布时间:2026-01-02 13:52:03

|

170人浏览过

|

来源于php中文网

原创

弗洛伊德算法必须以k为最外层循环,因状态转移依赖disti和distk在当前轮次已被正确更新;初始化需设disti=0、有向边权值、其余为安全INF(如0x3f3f3f3f),并判INF再更新以防溢出。

c++怎么实现弗洛伊德算法求最短路径_c++ 动态规划与邻接矩阵更新【案例】

弗洛伊德算法的核心是三重循环更新 dist[i][j]

它不依赖起点或终点预设,而是穷举所有中间点 k,检查是否通过 k 能让 i → j 更短。关键不是“怎么初始化”,而是“为什么必须按 k 为最外层循环”。如果把 ij 放最外层,dist[i][k]dist[k][j] 可能还没被当前轮次更新过,导致路径拼接失效。

初始化时,dist[i][i] = 0,有向边 (u, v, w) 对应 dist[u][v] = w,其余设为一个足够大的数(如 INT_MAX / 2,避免加法溢出)。

  • INT_MAX 直接参与加法易溢出,用 INT_MAX / 2 更安全
  • 图中含负权边可运行,但含负权环时,dist[i][i] 最终会为负 —— 这是检测负环的依据
  • 节点编号建议从 0 开始,避免数组越界和偏移计算错误

邻接矩阵更新要防止整型溢出和无效路径叠加

核心更新语句是:dist[i][j] = min(dist[i][j], dist[i][k] + dist[k][j])。但若 dist[i][k]dist[k][j] 仍是初始大值,直接相加会溢出。必须先判断有效性:

if (dist[i][k] != INF && dist[k][j] != INF) {
    dist[i][j] = min(dist[i][j], dist[i][k] + dist[k][j]);
}

否则哪怕逻辑上不可达,数值上也会因溢出变成极小负数,污染后续结果。

VIVA
VIVA

一个免费的AI创意视觉设计平台

下载

立即学习C++免费学习笔记(深入)”;

  • 不要用 == INT_MAX 判断“不可达”,浮点或大数运算后可能失真;统一用自定义常量 INF
  • 更新顺序不能交换:必须是 k 在最外层,否则不是 Floyd,而是错误的松弛顺序
  • 空间复杂度固定为 O(n²),无法像 Dijkstra 那样用堆优化;适合 n ≤ 500 的稠密图

动态规划视角下,dist[k][i][j] 的含义容易被误解

教材常说“dist[k][i][j] 表示只允许经过前 k 个节点时 i→j 的最短距离”,但这只是教学抽象。实际代码中根本没存三维数组 —— 它被压缩进二维并复用,靠 k 循环顺序保证状态转移正确。真正发生的是:每轮 k 结束后,dist[i][j] 已包含所有经节点 0..k 的最短路。

  • 没有显式维度 k,所以不能在循环中随意访问 “上一轮” 的 dist[i][j]
  • 不能把 dist[i][j] 更新写成 dist[i][j] = min(dist[i][j], dist[i][k] + dist[k][j]) 之外的形式(比如加条件跳过某些 k),会破坏 DP 状态定义
  • 若需输出路径,得额外维护 next[i][j] 记录中转点,每次更新 dist 时同步更新

常见报错:runtime error: signed integer overflow 或全为 INF

前者几乎全是没判 INF 就做加法;后者常见于输入没读进邻接矩阵(比如节点编号从 1 开始却往 [0][0] 写)、或忘了设 dist[i][i] = 0。还有一种隐蔽错误:用 memset(dist, 0x3f, sizeof dist) 初始化,但 0x3f3f3f3f 是常用 INF 值,而若误写成 0x7f7f7f7f,它接近 INT_MAX,加法极易溢出。

  • 推荐初始化方式:
    const int INF = 0x3f3f3f3f;
    for (int i = 0; i < n; i++) {
        for (int j = 0; j < n; j++) {
            dist[i][j] = (i == j) ? 0 : INF;
        }
    }
  • 调试时打印几组 dist[0][*]dist[*][n-1],比看全矩阵更高效
  • 算法本身不处理离散节点名(如字符串 ID),必须先映射为 0..n-1 整数索引
实际写的时候,最易忽略的是溢出防护和节点编号对齐——这两点出错,结果看起来“差不多”,但一换数据就崩。

相关专题

更多
java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1435

2023.10.24

php中三维数组怎样求和
php中三维数组怎样求和

php中三维数组求和的方法:1、创建一个php示例文件;2、定义一个名为“$total”的变量,用于记录累加的结果。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

91

2024.02.23

scripterror怎么解决
scripterror怎么解决

scripterror的解决办法有检查语法、文件路径、检查网络连接、浏览器兼容性、使用try-catch语句、使用开发者工具进行调试、更新浏览器和JavaScript库或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

184

2023.10.18

500error怎么解决
500error怎么解决

500error的解决办法有检查服务器日志、检查代码、检查服务器配置、更新软件版本、重新启动服务、调试代码和寻求帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

265

2023.10.25

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

250

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

205

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1435

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

609

2023.11.24

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

74

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 5.8万人学习

C 教程
C 教程

共75课时 | 3.8万人学习

C++教程
C++教程

共115课时 | 10.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号