0

0

C++怎么实现Dijkstra算法_C++图论最短路径逻辑【实战】

穿越時空

穿越時空

发布时间:2026-03-13 11:41:03

|

368人浏览过

|

来源于php中文网

原创

应使用 std::priority_queue 而非 std::set 或手写堆,因其支持 O(log n) 入队和 O(1) 取顶、常数小;虽不支持减键,但插入新记录+忽略过期项更简洁安全;须用 std::greater 声明为小根堆,格式为 std::priority_queue<std::pair<int,int>, std::vector<std::pair<int,int>>, std::greater<>> pq;

c++怎么实现dijkstra算法_c++图论最短路径逻辑【实战】

为什么用 std::priority_queue 而不是 std::set 或手写堆

因为 Dijkstra 的核心是反复取当前最小距离节点,std::priority_queue 天然支持 O(log n) 入队和 O(1) 取顶,且常数小。虽然它不支持修改已有元素的优先级(即“减键”操作),但直接插入新记录 + 忽略过期记录,比用 std::set 维护迭代器或手写可删堆更简洁、不易出错。

常见错误现象:std::priority_queue 默认是大根堆,必须传入 std::greater;否则第一次弹出的就是最大距离,算法立刻失效。

  • 声明方式必须是:std::priority_queue<:pair int>, std::vector<:pair int>>, std::greater> pq;</:pair></:pair>(第一个 int 是距离,第二个是节点编号)
  • 不要试图在队列里更新已存在节点的距离——直接 pq.push({new_dist, v}),后续遇到旧记录时靠 if (dist[v] 跳过
  • 如果图稀疏(边数 m ≈ n),这种“懒删除”几乎无性能损失;稠密图也只多 log n 级别冗余操作

dist 数组初始化为 INT_MAX 还是 1e9+7

INT_MAX 有溢出风险:当某条边权很大(比如接近 INT_MAX),松弛时 dist[u] + w 可能溢出为负数,导致错误更新。实际项目中建议用一个明显大于所有可能最短路径的常量,比如 1e9+100x3f3f3f3f(约 10.6 亿,且满足 0x3f3f3f3f * 2 )。

  • 初始化写法:std::vector<int> dist(n, 0x3f3f3f3f); dist[start] = 0;</int>
  • 判断是否不可达:用 if (dist[v] == 0x3f3f3f3f),而不是 == INT_MAX
  • 如果边权含负数,Dijkstra 本身就不适用——此时应换 SPFA 或 Bellman-Ford,别硬改初始化值

邻接表存图时,vector<vector int>>></vector>vector<vector>></vector> 哪个更稳妥

前者更轻量、通用性强;后者适合需要扩展字段(如边 ID、方向标记)的场景。但绝大多数 Dijkstra 实战只要“终点 + 权重”,用 pair 就够了,且避免结构体构造/拷贝开销。

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

Mokker AI
Mokker AI

AI产品图添加背景

下载

容易踩的坑:用 vector<vector>></vector> 存两个字段(比如 {v, w}),看似省事,但读取时易写错下标(g[u][i][0] vs g[u][i][1]),可读性差,编译也不报错。

  • 推荐写法:vector<vector int>>> g(n);</vector>,加边:g[u].emplace_back(v, w);
  • 遍历时清晰:for (auto [v, w] : g[u]) { ... }(C++17 结构化绑定)
  • 如果必须用结构体,至少定义 struct Edge { int to, w; };,别裸用 vector<int></int>

单源最短路径跑完后,怎么还原具体路径

Dijkstra 本身不记录路径,得额外维护 prev 数组。每次成功松弛 dist[v] 时,同步记录 prev[v] = u。注意:只在 if (dist[u] + w 成立时更新,不是每次访问都写。

还原路径时从终点倒推,用栈或 vector 反转,别忘了起点本身也要加入结果。

  • 初始化:vector<int> prev(n, -1);</int>,起点的 prev[start] = start 或保持 -1 都行,但还原逻辑要一致
  • 松弛时加一句:if (dist[u] + w
  • 还原示例:vector<int> path; for (int x = end; x != -1; x = prev[x]) path.push_back(x); reverse(path.begin(), path.end());</int>

复杂点在于:路径还原和距离计算是解耦的,很多人只验证了 dist[end] 正确就以为路径也对了,其实 prev 更新漏条件或初始化错,会导致路径乱序或截断。检查时最好挑一个中间节点,手动核对它的 prev 是否指向真正前驱。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
edge是什么浏览器
edge是什么浏览器

Edge是一款由Microsoft开发的网页浏览器,是Windows 10操作系统中默认的浏览器,其目标是提供更快、更安全、更现代化的浏览器体验。本专题为大家提供edge浏览器相关的文章、下载、课程内容,供大家免费下载体验。

1734

2023.08.21

IE浏览器自动跳转EDGE如何恢复
IE浏览器自动跳转EDGE如何恢复

ie浏览器自动跳转edge的解决办法:1、更改默认浏览器设置;2、阻止edge浏览器的自动跳转;3、更改超链接的默认打开方式;4、禁用“快速网页查看器”;5、卸载edge浏览器;6、检查第三方插件或应用程序等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

397

2024.03.05

如何解决Edge打开但没有标题的问题
如何解决Edge打开但没有标题的问题

若 Microsoft Edge 浏览器打开后无标题(窗口空白或标题栏缺失),可尝试以下方法解决: 重启 Edge:关闭所有窗口,重新启动浏览器。 重置窗口布局:右击任务栏 Edge 图标 → 选择「最大化」或「还原」。 禁用扩展:进入 edge://extensions 临时关闭插件测试。 重置浏览器设置:前往 edge://settings/reset 恢复默认配置。 更新或重装 Edge:检查最新版本,或通过控制面板修复

1038

2025.04.24

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

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

1567

2023.10.24

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

847

2023.08.22

java break和continue
java break和continue

本专题整合了java break和continue的区别相关内容,阅读专题下面的文章了解更多详细内容。

261

2025.10.24

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

490

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

202

2025.07.04

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

3

2026.03.13

热门下载

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

精品课程

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

共94课时 | 11.2万人学习

C 教程
C 教程

共75课时 | 5.4万人学习

C++教程
C++教程

共115课时 | 21.6万人学习

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

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