news 2026/4/16 15:50:33

Dijkstra - 单源最短路径

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Dijkstra - 单源最短路径

算法:Dijkstra [堆优化(优先队列)]

求解:单源最短路径

核心思想:贪心,每次从未确定最短距离的节点中,选择距离源点最近的节点,用该节点更新其邻接点的距离。

这是一个堆优化的Dijkstra最短路径算法实现。让我为您详细解析每个部分:

一、数据结构解析

1. 邻接表存储(链式前向星)

邻接表存储图

int h[1005],e[1005],ne[1005],idx,w[1005];//邻接表存储图

2. 优先队列(最小堆)

优先队列 q 用于存储 {距离, 节点} 对

greater<pair<int,int>> 确保队列按距离从小到大排序

priority_queue<pair<int,int>,vector<pair<int,int>>,greater<pair<int,int>>> q;

3. 辅助数组

bool s[1005];//标记数组,记录节点是否已确定最短距离 int dis[1005];//存储源点到各节点的最短距离
  • dis[1005]: 源点到各点的最短距离

  • s[1005]: 标记节点是否已确定最短路径

二、核心算法流程

1.输入

memset(h,-1,sizeof(h)); cin>>n>>m; for(int i=1; i<=m; i++) { int a,b,c; cin>>a>>b>>c; if(c<0) return 0; add(a,b,c); }

同时,加边函数:

void add(int a,int b,int c)//邻接表的边添加函数,用于构建图的邻接表存储结构 { e[idx]=b;//终点 ne[idx]=h[a];//下一条边的索引 w[idx]=c;//边的权值 h[a]=idx++;//更新头节点的下一条边索引 }

2. 初始化

memset(dis,0x3f,sizeof(dis));//初始化距离为无穷大 dis[1]=0;//源点到自身距离为0 q.push({0,1});//源点入队

2. 主循环

while(!q.empty()) { auto t=q.top();//取出距离源点最近的节点 q.pop(); if(s[t.second]) continue;//如果该节点已确定最短距离,则跳过 s[t.second]=1;//标记该节点为已确定 for(int i=h[t.second]; i!=-1; i=ne[i])//遍历该节点的所有邻接边 { if(dis[e[i]]>t.first+w[i])//如果通过当前节点到达邻接节点的距离更短 { dis[e[i]]=t.first+w[i];//更新距离 q.push({dis[e[i]],e[i]});//邻接节点入队 } } }

4.输出

if(dis[n]==0x3f3f3f3f) cout<<-1;//如果目标节点不可达,输出-1 else cout<<dis[n];//输出源点到目标节点的最短距离

三、关键代码详解

边添加函数add()

示例:添加边 1→2(3),1→3(5)

初始:h[1] = -1 添加1→2(3):e[0]=2, ne[0]=-1, h[1]=0, idx=1 添加1→3(5):e[1]=3, ne[1]=0, h[1]=1, idx=2 结果形成链表:1 → 边1(1→3) → 边0(1→2) → -1

四、算法特性

时间复杂度

  • 普通Dijkstra: O(n²) - 适合稠密图

  • 堆优化Dijkstra: O((n+m)log n) - 适合稀疏图(本实现)

空间复杂度

  • O(n + m):存储图和距离数组

限制条件

  1. 不能处理负权边(会陷入死循环或得到错误结果)

  2. 适用于有向图和无向图(本代码实现的是有向图

  3. 需要非负权值

五、输入输出示例

输入格式

n m a1 b1 c1 a2 b2 c2 ... am bm cm

示例

4 5 1 2 2 1 3 5 2 3 1 2 4 4 3 4 3

计算过程:

初始:dis = [0, ∞, ∞, ∞] 第1轮:节点1出队,更新dis[2]=2, dis[3]=5 第2轮:节点2出队,更新dis[3]=3, dis[4]=6 第3轮:节点3出队,更新dis[4]=6(不变) 结果:dis[4] = 6

六、算法优化点(AI所说)

1. 懒惰删除技巧

if(s[t.second]) continue; // 避免重复处理

同一个节点可能多次入队,但只处理第一次出队的情况

2. 无穷大表示

0x3f3f3f3f ≈ 1.06×10⁹

使用这个值作为无穷大,两个无穷大相加不会溢出

3. 堆优化的优势

相比普通Dijkstra的O(n²),堆优化:

  • 减少查找最小距离节点的时间

  • 适合边数远小于n²的稀疏图

七、常见问题

1. 对于负权边

Dijkstra基于贪心策略,假设"当前最小距离就是最终最短距离"。负权边会破坏这个假设。

2. 对于无向图

添加边时调用两次:

add(a,b,c); add(b,a,c);

3. 源点不是节点1:

只需修改初始化:

dis[source]=0; q.push({0,source}); //source是源点

写得好就关注,点赞一下!

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/16 12:16:57

HarmonyOS配置文件终极指南:从入门到精通的完整教程

HarmonyOS配置文件终极指南&#xff1a;从入门到精通的完整教程 【免费下载链接】harmony-utils harmony-utils 一款功能丰富且极易上手的HarmonyOS工具库&#xff0c;借助众多实用工具类&#xff0c;致力于助力开发者迅速构建鸿蒙应用。其封装的工具涵盖了APP、设备、屏幕、授…

作者头像 李华
网站建设 2026/4/16 11:56:07

通天之分组背包(洛谷P1757 )

题目背景直达通天路小 A 历险记第二篇题目描述自 01 背包问世之后&#xff0c;小 A 对此深感兴趣。一天&#xff0c;小 A 去远游&#xff0c;却发现他的背包不同于 01 背包&#xff0c;他的物品大致可分为 k 组&#xff0c;每组中的物品相互冲突&#xff0c;现在&#xff0c;他…

作者头像 李华
网站建设 2026/4/16 11:55:01

Ruby爬虫框架Wombat:用优雅DSL轻松提取结构化数据

Ruby爬虫框架Wombat&#xff1a;用优雅DSL轻松提取结构化数据 【免费下载链接】awesome-crawler A collection of awesome web crawler,spider in different languages 项目地址: https://gitcode.com/gh_mirrors/aw/awesome-crawler 还在为网页数据提取而烦恼吗&#x…

作者头像 李华
网站建设 2026/4/16 11:55:19

WinCC 7.4 完整安装指南与资源获取

WinCC 7.4 完整安装指南与资源获取 【免费下载链接】WinCC7.4安装包下载 本仓库提供SIMATIC WINCC 7.4 安装包的完整版下载。该安装包包含了WinCC 7.4的所有必要组件&#xff0c;适用于需要安装或升级WinCC 7.4的用户 项目地址: https://gitcode.com/Open-source-documentati…

作者头像 李华
网站建设 2026/4/16 7:22:10

Citybound道路系统完整指南:5步掌握智能路网设计技巧

Citybound道路系统完整指南&#xff1a;5步掌握智能路网设计技巧 【免费下载链接】citybound A work-in-progress, open-source, multi-player city simulation game. 项目地址: https://gitcode.com/gh_mirrors/ci/citybound Citybound道路系统是这款开源多玩家城市模拟…

作者头像 李华
网站建设 2026/4/16 10:39:56

终极hekate快捷启动指南:3分钟实现Switch一键直达

还在为Nintendo Switch每次启动时繁琐的系统选择而烦恼吗&#xff1f;传统启动方式不仅耗时费力&#xff0c;还容易在关键时刻选错选项。本指南将为你展示hekate快捷启动的实用技巧&#xff0c;让你告别重复操作&#xff0c;轻松实现一键直达常用系统或工具。 【免费下载链接】…

作者头像 李华