news 2026/5/7 23:25:45

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/5/8 12:33:34

亲手搭建原子级观测设备:OpenSTM终极指南

亲手搭建原子级观测设备&#xff1a;OpenSTM终极指南 【免费下载链接】OpenSTM OpenSTM - 一个扫描隧道显微镜项目&#xff0c;可能用于科研或精密工程领域。 项目地址: https://gitcode.com/gh_mirrors/op/OpenSTM 想要亲眼看到原子的排列吗&#xff1f;现在&#xff0…

作者头像 李华
网站建设 2026/4/30 23:07:21

突破性进展:NVIDIA OpenReasoning推理模型重塑AI编程新范式

突破性进展&#xff1a;NVIDIA OpenReasoning推理模型重塑AI编程新范式 【免费下载链接】OpenReasoning-Nemotron-14B 项目地址: https://ai.gitcode.com/hf_mirrors/nvidia/OpenReasoning-Nemotron-14B 在人工智能与编程深度融合的时代背景下&#xff0c;NVIDIA最新推…

作者头像 李华
网站建设 2026/5/1 0:18:44

Qwen3 Embedding模型终极指南:vLLM Ascend快速部署与性能调优

在人工智能语义理解领域&#xff0c;Qwen3 Embedding模型系列以其卓越的多语言能力和灵活的向量表示&#xff0c;为文本检索与重排序任务带来了革命性突破。本指南将带您深度探索基于vLLM Ascend部署这一前沿技术的完整流程。 【免费下载链接】Qwen3-Reranker-8B 项目地址: …

作者头像 李华
网站建设 2026/4/30 23:37:28

5个步骤掌握LXGW Neo XiHei:从下载到专业应用的完整指南

LXGW Neo XiHei&#xff08;霞鹜新晰黑&#xff09;是一款基于日本IPAexGothic改造的中文开源黑体字体&#xff0c;专为现代数字环境设计。这款开源字体不仅保留了日文字体的优雅气质&#xff0c;还针对中文使用习惯进行了全面优化&#xff0c;支持2.2万汉字和多种语言&#xf…

作者头像 李华
网站建设 2026/5/8 7:24:22

Apache ECharts教育数据可视化终极指南:从零到精通的完整方案

Apache ECharts教育数据可视化终极指南&#xff1a;从零到精通的完整方案 【免费下载链接】echarts Apache ECharts is a powerful, interactive charting and data visualization library for browser 项目地址: https://gitcode.com/gh_mirrors/echarts16/echarts 在当…

作者头像 李华