【图论】迪杰特斯拉算法

【图论】迪杰特斯拉算法

文章目录

在这里插入图片描述

迪杰特斯拉算法

迪杰特斯拉算法是由荷兰计算机科学家艾兹赫尔·迪杰特斯拉(Edsger W. Dijkstra)在1956年提出的,用于解决单源最短路径问题的经典算法。该算法的目标是从一个起始顶点找到到图中其他顶点的最短路径。

主要特点

  • 适用于带权图,其中权重为非负数。(为什么只适用于非负数,因为迪杰斯特拉的思想是贪心测量,当有负权引入的时候,贪心策略将不再适用)
  • 解决从单个源点到其他所有顶点的最短路径问题。
  • 时间复杂度:当使用优先队列(例如堆)时,复杂度为 O ( E log ⁡ V ) O(E \log V) O(ElogV),其中 V V V 为顶点数量, E E E 为边的数量。

基本思想

Dijkstra算法通过不断探索距离最近的顶点,逐步扩展其最短路径的已知范围,直到找到从源点到所有其他顶点的最短路径。该算法基于贪心策略:每一步选择尚未处理的、距离源点最近的顶点进行扩展。

算法步骤

  1. 初始化
    • 将起始顶点的距离设为0,其余所有顶点的距离设为∞(表示不可达)。
    • 使用一个优先队列(或最小堆)来存储顶点及其当前的最短距离。
  2. 取距离源点最近的顶点,并标记为已处理。
  3. 对于该顶点的每个邻接顶点,更新其最短距离(如果通过当前顶点可以获得更短的路径,则更新距离)。
  4. 重复步骤2和3,直到所有顶点都被处理过,或优先队列为空。

示例

在这里插入图片描述

实现迪杰斯特拉算法

基本步骤

首先假设我们有如下的一个图:

在这里插入图片描述


灰色的点代表起点,我们需要从起点开始算每个点到起点的最短路径。
第一步按照迪杰斯特拉的表述应该将起点到起点的最短路径初始化为0,起点到另外其他点的距离初始化为正无穷。

在这里插入图片描述


这里我们更新完s点之后,应该更新和s点相连的点的最短路径了,由于之前s到t点和到y点的最短路径是正无穷,由于st和sy都小于两个最短路径,所以更新两个最短路径。

在这里插入图片描述


根据贪心策略,更新出来的两个最短路径,sy更小,所以我们应该更新y相连的路径。

在这里插入图片描述


所以接下来我们应该更新y连接的点的最短路径。

在这里插入图片描述


由于原本s到t的最短路径是10,但是现在的最短路径更新了之后是8,所以更新结果,由于之前s到x的最短路径是正无穷,所以现在将最短路径更新为14,s到z 也是相同的,因为之前的最短路径是正无穷,所以现在将最短路径更新为7.
在这三条最短路径中选择最短的那条:

在这里插入图片描述


这里就应该以z为新的起点

在这里插入图片描述


更新z连接的顶点,z一共有两条边,一条是zs,一条是zx,由于s到s是最近的0,所以这里不需要更新,由于之前s到x的距离是14,所以这里更新s到x的距离。
接下来再从剩下的边中,选择最小的路径。

在这里插入图片描述


从t点开始只有一条路径,所以这里t到x,tx是1,由于之前的st的最短路径是8,所以此时的最短路径是9,9<13所以这里应该更新最短路径为9

在这里插入图片描述


这里最短路径算是更新完了。

算法思路

由于这里我们涉及到最短路径,所以应该开辟一个dist数组,我们来想一想一个dist数组是否能解决问题,很显然是不能的,我们还需要一个数组,记录当前最短路径的前一个顶点的下标,在遍历的时候我们可以索引每一个顶点了。
代码展示:

voidDijkstra(const V& src, vector<W>& dist, vector<int>& pPath){//获取起点的下标size_t srci =GetVertexIndex(src);//确定节点的数量size_t n = _vertexs.size(); dist.resize(n, MAX_W); pPath.resize(n, MAX_W);//自己到自己的距离是0 dist[srci]=0;//自己的前一个是自己 pPath[srci]= srci;//已经确定最短路径的顶点的集合 vector<bool>S(n, false);for(size_t j=0;j<n;j++){//选择最短路径顶点且不在s中更新其他路径int u =0;//最小的那个点 W min = MAX_W;//最小权值for(size_t i =0;i < n;i++){if(S[i]== false && dist[i]< min){//选出最小的点 u = i;//更新最小权值 min = dist[i];}}//u被选出来 S[u]= true;//松弛更新u连接出去的顶点v srci->u u->vfor(size_t v =0;v < n;v++){//确定u连接出去的所有边if(S[v]== false && _matrix[u][v]!= MAX_W &&(dist[u]+ _matrix[u][v])< dist[v]){ dist[v]= dist[u]+ _matrix[u][v];//将v的父亲更新为u pPath[v]= u;}}}}

打印路径节点和最短路径:

//打印最短路径voidPrinrtShotPath(const V& src, vector<W>& dist, vector<int>& pPath){//获取起点的下标size_t srci =GetVertexIndex(src);//确定节点的数量size_t n = _vertexs.size();for(size_t i =0;i < n;i++){if(i != srci){// 先找出i顶点的路径 vector<int> path;size_t parenti = i;//先将自己存进去(自己不是原点先push进去)while(parenti != srci){ path.push_back(parenti); parenti = pPath[parenti];} path.push_back(srci);//将路径逆置reverse(path.begin(), path.end());//打印出路径for(auto e : path){ cout << _vertexs[e]<<"->";}//打印最短路径 cout << dist[i]; cout << endl;}}}

因为根据最后一个节点去推上一个节点,推完之后数组中的节点会是一个反着的路径,所以在打印的时候,应该先把数组逆置,逆置之后再进行打印。

总结

在本文中,我们深入探讨了迪杰斯特拉算法的原理与应用。作为一种经典的最短路径算法,迪杰斯特拉算法通过优先队列有效地解决了从单一源点到其他所有节点的最短路径问题。我们分析了其时间复杂度和空间复杂度,了解了在不同图形结构下的性能表现。

通过示例和实现,我们不仅掌握了算法的基本步骤,还体验了其在实际应用中的重要性。无论是在交通导航、网络路由还是各种优化问题中,迪杰斯特拉算法都发挥着不可或缺的作用。

希望本文能够帮助你更好地理解迪杰斯特拉算法,并为你在图论和算法领域的进一步学习打下坚实的基础。如果你有任何疑问或想法,欢迎在评论区与我交流!

Read more

Flutter 三方库 bones_ui 的鸿蒙化适配指南 - 打造直观、响应式的 Web 风格 UI 交互体验

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 三方库 bones_ui 的鸿蒙化适配指南 - 打造直观、响应式的 Web 风格 UI 交互体验 Flutter for OpenHarmony 开发者在构建具有 Web 质感的跨平台应用时,UI 框架的选择至关重要。本文将带大家深度调研 Dart 三方库 bones_ui 在鸿蒙系统上的适配方案,探索如何利用其直观的组件架构,加速鸿蒙桌面级应用的开发效率。 前言 在移动端和桌面端融合的今天,开发者往往希望一套代码能同时适配多种屏幕形态。bones_ui 原生为 Dart Web 打造,但在 Flutter for OpenHarmony 的大前端生态中,其简洁的 UI 组件设计思想对我们构建鸿蒙跨平台应用具有极大的参考价值。

By Ne0inhk
35道常见的前端vue面试题,零基础入门到精通,收藏这篇就够了

35道常见的前端vue面试题,零基础入门到精通,收藏这篇就够了

来源 | https://segmentfault.com/a/1190000021936876 今天这篇文章给大家分享一些常见的前端vue面试题。有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。 对于前端来说,尽管css、html、js是主要的基础知识,但是随着技术的不断发展,出现了很多优秀的mv*框架以及小程序框架。因此,对于前端开发者而言,需要对一些前端框架进行熟练掌握。这篇文章我们一起来聊一聊VUE及全家桶的常见面试问题。 1、请讲述下VUE的MVVM的理解? MVVM 是 Model-View-ViewModel的缩写,即将数据模型与数据表现层通过数据驱动进行分离,从而只需要关系数据模型的开发,而不需要考虑页面的表现,具体说来如下: Model代表数据模型:主要用于定义数据和操作的业务逻辑。 View代表页面展示组件(即dom展现形式):负责将数据模型转化成UI 展现出来。 ViewModel为model和view之间的桥梁:监听模型数据的改变和控制视图行为、处理用户交互。通过双向数据绑定把 View 层和 Model 层连接了起来,而View

By Ne0inhk
【超详细】DEIM:最强实时目标检测算法-Visdrone2019无人机数据集实战

【超详细】DEIM:最强实时目标检测算法-Visdrone2019无人机数据集实战

主要内容如下: 一、论文解析 二、基于DEIM-D-FINE-S训练Visdrone2019无人机数据集 1、Visdrone2019数据集介绍 2、模型训练、验证及测试 3、onnx导出与测试 4、与YOLOv8\11进行结果对比 服务器:NVIDIA RTX4090 24G 运行环境:Python=3.8(要求>=3.8),torch2.3.1+cu121(要求>=2.0.1) Visdrone2019-COCO格式数据集百度AI stduio下载链接:https://aistudio.baidu.com/datasetdetail/226107/0 Visdrone-YOLO格式数据集下载链接:https://aistudio.baidu.com/

By Ne0inhk