单源最短路径问题--Dijkstra算法

2024-04-17 13:28

本文主要是介绍单源最短路径问题--Dijkstra算法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Dijkstra算法实现

  • 1.算法过程
  • 2.代码实现过程
    • 2.1朴素Dijkstra实现
    • 2.2朴素Dijkstra实现

1.算法过程

Dijkstra算法是图论中用来解决单元最短路径的算法,我们通过将点划分成为两个部分,一个是已经确定了源点到达此处最短路径的点,另外一个部分是还不确定最短路径的点。我们每次每一次选择状态图中最短路,将最短路端点A标记成为已经确定最短路径的点集,接着看看是否有通过A这个最短路径可以有一条更短的路到达未确定的点,如果有的话,就更新源点到达这个点路径的最短信息。接着重复找到最短的路径加入,重复上面步骤,指导所有点都找完。确切算法描述过程如下:
在这里插入图片描述在算法实现过程中,每次选择最短路径和更新路径信息是耗时较多的地方,所以我们选择最短路的时候,是否通过堆来优化,将Dijkstra算法划分成为了朴素Dijkstra(就是没次选择最短路加入的时候,需要将所有路径长度遍历一次)和堆优化的Dijkstra。

2.代码实现过程

这里以leecode中题目为例子来讲述Dijkstra的实现过程,代码是参考灵茶山艾府的题解
在这里插入图片描述

2.1朴素Dijkstra实现

注意朴素Dijkstra的实现利用的图的存储结构是邻接矩阵

class Solution {
public:int networkDelayTime(vector<vector<int>>& times, int n, int k) {vector<vector<int>> g(n, vector<int>(n, INT_MAX / 2));//邻接矩阵存储for(auto &t : times){g[t[0] - 1][t[1] - 1] = t[2];}//dis用来存储单元的最短的路径;done用来标记是否已经确定了最短路径vector<int> dis(n, INT_MAX / 2), done(n);//顶点k作为源点dis[k - 1] = 0;while(true){int x = -1;for(int i = 0; i < n; i++){//用来确定最短路径:x是用来标记确定最短路径的另一端的点if(!done[i] && (x < 0 || dis[i] < dis[x])){x = i;}}//表示已经确定了所有点的最短路径if(x < 0) return ranges::max(dis);//有节点无法到达if(dis[x] == INT_MAX / 2) return -1;//最短路长度已经确定,无法变得更小done[x] = true;//确定源点到达x最短路径之后,需要判断是否更新源点到达x相连点的路径长度for(int y = 0; y < n; y++){//更新x的邻居的最短路dis[y] = min(dis[y], dis[x] + g[x][y]);}}}
};

2.2朴素Dijkstra实现

堆优化实现的过程中,需要去把这个判断语句的作用给弄明白

if(dx > dis[x]) continue;

这个因为我们利用堆实现的过程中,我们确定了最短路径的点加入之后,我们的更新路径的信息过程不会直接去把堆中的信息给修改,而是直接再插入一个更短的路径信息上去。
例如下图:我们源点确定了之后,会将源点相连的点1和点2的路径信息更新为3和10(其余点为无穷),
但是接下我们堆中的最短路径长度3(源点到达点1),此时通过点1我们发现源点到达点2的距离可以有一条比长度10更短的路径长度7,但是此时我们不会直接去将堆中的10给删除,而是插入一个7。同时记住一下到达点2的路可以用7就到达,所以之后我们堆弹出10的时候,我们不会拿着10(脏数据)去更新点2的最短路径。
在这里插入图片描述

class Solution {
public://邻接表适合稀疏图int networkDelayTime(vector<vector<int>>& times, int n, int k) {vector<vector<pair<int, int>>> g(n);//邻接表for(auto &t :times){//x -> y: 把y和x到y距离打包成pair拉链到点x下面g[t[0] - 1].emplace_back(t[1] - 1, t[2]);}//存储最短路径vector<int> dis(n, INT_MAX);//从源点k出发dis[k - 1] = 0;//定义一个最小堆,维护源点到达其他顶点未选择路径的最短路(第二个参数指名了底层实现的堆的数据结构)priority_queue<pair<int, int>, vector<pair<int, int>>, greater<>> pq;//pq中的pair为:<distance, point>pq.emplace(0, k - 1);while(!pq.empty()){auto [dx, x] = pq.top();pq.pop();//x之前出堆过(对于同一个 x,入堆的 dis[x] 只会越来越小,那么首先弹出的就是最小的 dis[x],//后面弹出的 dx 必定比 dis[x] 大。既然首先弹出的是最小的,所以首次弹出的 dx 等于 dis[x])if(dx > dis[x]) continue;for(auto &[y, d] :g[x]){int new_dis = dx + d;if(new_dis < dis[y]){//跟新x的邻居的最短路dis[y] = new_dis;pq.emplace(new_dis, y);}}}int mx = ranges::max(dis);return mx < INT_MAX ? mx : -1;}
};

这篇关于单源最短路径问题--Dijkstra算法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/911914

相关文章

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

Java中的雪花算法Snowflake解析与实践技巧

《Java中的雪花算法Snowflake解析与实践技巧》本文解析了雪花算法的原理、Java实现及生产实践,涵盖ID结构、位运算技巧、时钟回拨处理、WorkerId分配等关键点,并探讨了百度UidGen... 目录一、雪花算法核心原理1.1 算法起源1.2 ID结构详解1.3 核心特性二、Java实现解析2.

Redis出现中文乱码的问题及解决

《Redis出现中文乱码的问题及解决》:本文主要介绍Redis出现中文乱码的问题及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1. 问题的产生2China编程. 问题的解决redihttp://www.chinasem.cns数据进制问题的解决中文乱码问题解决总结

全面解析MySQL索引长度限制问题与解决方案

《全面解析MySQL索引长度限制问题与解决方案》MySQL对索引长度设限是为了保持高效的数据检索性能,这个限制不是MySQL的缺陷,而是数据库设计中的权衡结果,下面我们就来看看如何解决这一问题吧... 目录引言:为什么会有索引键长度问题?一、问题根源深度解析mysql索引长度限制原理实际场景示例二、五大解决

Springboot如何正确使用AOP问题

《Springboot如何正确使用AOP问题》:本文主要介绍Springboot如何正确使用AOP问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录​一、AOP概念二、切点表达式​execution表达式案例三、AOP通知四、springboot中使用AOP导出

Python中Tensorflow无法调用GPU问题的解决方法

《Python中Tensorflow无法调用GPU问题的解决方法》文章详解如何解决TensorFlow在Windows无法识别GPU的问题,需降级至2.10版本,安装匹配CUDA11.2和cuDNN... 当用以下代码查看GPU数量时,gpuspython返回的是一个空列表,说明tensorflow没有找到

解决未解析的依赖项:‘net.sf.json-lib:json-lib:jar:2.4‘问题

《解决未解析的依赖项:‘net.sf.json-lib:json-lib:jar:2.4‘问题》:本文主要介绍解决未解析的依赖项:‘net.sf.json-lib:json-lib:jar:2.4... 目录未解析的依赖项:‘net.sf.json-lib:json-lib:jar:2.4‘打开pom.XM

IDEA Maven提示:未解析的依赖项的问题及解决

《IDEAMaven提示:未解析的依赖项的问题及解决》:本文主要介绍IDEAMaven提示:未解析的依赖项的问题及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝... 目录IDEA Maven提示:未解析的依编程赖项例如总结IDEA Maven提示:未解析的依赖项例如

Redis分片集群、数据读写规则问题小结

《Redis分片集群、数据读写规则问题小结》本文介绍了Redis分片集群的原理,通过数据分片和哈希槽机制解决单机内存限制与写瓶颈问题,实现分布式存储和高并发处理,但存在通信开销大、维护复杂及对事务支持... 目录一、分片集群解android决的问题二、分片集群图解 分片集群特征如何解决的上述问题?(与哨兵模

SpringBoot+Redis防止接口重复提交问题

《SpringBoot+Redis防止接口重复提交问题》:本文主要介绍SpringBoot+Redis防止接口重复提交问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不... 目录前言实现思路代码示例测试总结前言在项目的使用使用过程中,经常会出现某些操作在短时间内频繁提交。例