当前位置: 首页 > news >正文

最短路径之Dijkstra算法

2019独角兽企业重金招聘Python工程师标准>>> hot3.png

今天为大家分享的算法是为解决最短路径算法的Dijkstra算法(简称D算法),这是一个解决从点到点之间最短路径的问题,看下面这张图:

每天一个算法——最短路径之Dijkstra算法

这里,我们想要得出节点a(节点1)到节点b(节点5)的最短路径,就是怎么走可以使得权重值的和最小,每一条边都有一个权重。

今天我们介绍的D算法就是解决这类问题的,这是一种贪心算法,每次只取权重和最小的点,通过不断加入节点,来更新源节点a到各个节点的最短路径,直到所有节点遍历完。

算法步骤:

1、定义,遍历过的节点集合为S,集合U为其余节点(即未遍历)。初始时,S只包含源点v,即S={v},v的距离为0。U包含除v外的其他顶点,即:U={其余顶点}。若v与U中顶点u有边,则<u,v>正常有权值,若u不是v的出边邻接点,则<u,v>权值为∞。

注:这里集合S、U,是为了判断哪些节点已经遍历过,如果U为空了,就不继续执行。

2、从集合U中选取一个距离v最小的顶点k,把k加入到S中。

3、以k为新考虑的中间点,修改v到U中各顶点的距离;若从源点v到顶点w的距离(经过顶点k)比原来距离(不经过顶点k)短,则修改v到w的距离值。

例子:

每天一个算法——最短路径之Dijkstra算法

4、重复步骤2、3直到所有顶点都包含在S中。

上面就是D算法的处理步骤,可能大家第一次看和我一样很迷茫,不要紧,我们结合上面这个图,使用D算法来详细介绍每个步骤:

1、初始化步骤

用一个一维数组DIS来表示节点1到各个节点的最短路径(即权重),没有连线的用∞表示。除此之外,为了防止节点重复计算,我们把节点分成两组,一组已经遍历的节点集合S,另一组还没遍历集合U。初始化的时候,节点1在集合S中。

每天一个算法——最短路径之Dijkstra算法

注:节点1到自己的权重为0。

2、从集合U中获取离节点1最近的点(参考U在数组中的最小值,是节点2),加入到集合S,并重新计算DIS数组。

(1)节点2有到节点3和4的边,所以数组DIS的3和4位置的值可能会变动。

(2)2到3的权重为10,所以1到3的权重为17(7+10),17大于9,所以不用变动。

(3)3到4的权重为15,所以1到4的权重为22(7+15),22小于无穷,所以DIS[4]=22。(这里数组角标大家注意下,DIS[4]表示第四个位置,起始位置为1)

每天一个算法——最短路径之Dijkstra算法

3、重复步骤2,获取U里面距离最近的点(这里为节点3),重新计算DIS数组。

(1)节点3这里和4、6有边,所以DIS的位置4、6可能需要修改值。(节点2在S中,只考虑U中没遍历过的节点)。

(2)3到4的权重为11,所以1到4的权重为20(9+11),小于22(DIS[4]),所以DIS[4]=20。

(3)3到6的权重为2,所以1到6的权重为11(9+2),小于14(DIS[6]),所以DIS[6]=11。

每天一个算法——最短路径之Dijkstra算法

4、重复步骤2,取节点6加入S,重新计算DIS。

节点6只有到5的边了,所以只修改DIS[6]的值。这里节点1到节点5的权重为20(11+9),小于无穷(DIS[5]),所以DIS[5]=20。

每天一个算法——最短路径之Dijkstra算法

5、继续重复。

这次获取到节点4,从DIS数组可以知道1到4的权重(20)已经大于等于1到5的权重(20),所以无论如何也无法从节点4取到权重更小的路径了,所以可以舍弃(D算法是无法解决负权重问题,所以图的权重必须为正)。

 

由于节点1到节点5没有边连接,所以权重为无穷,大于20。所以,算法的最终结果就是:

每天一个算法——最短路径之Dijkstra算法

节点1到节点5的最短路径是20,

顺序是1->3->6->5。

有了算法,必须要有代码才有说服力,这里我用C语言实现了D算法的代码,大家有兴趣慢慢看,慢慢研究。我贴的是部分代码,其他不重要代码省略。

预定义变量:

每天一个算法——最短路径之Dijkstra算法

数据初始化:

每天一个算法——最短路径之Dijkstra算法

D算法具体逻辑方法:

每天一个算法——最短路径之Dijkstra算法

运行结果:

每天一个算法——最短路径之Dijkstra算法

转载于:https://my.oschina.net/u/2391658/blog/898513

相关文章:

  • nyoj 14 会场安排问题
  • 使用Fetch
  • padding的使用
  • 【extjs6学习笔记】0.2 准备:类库结构
  • 欢迎报名第三届中国移动互联网测试开发大会
  • python+selenium自动化测试(六)
  • The based of tuning
  • 《Spring 5 官方文档》5. 验证、数据绑定和类型转换(一)
  • oracle DDL
  • 一般造成Linux系统死机的原因
  • 一段PHP异常
  • 从零搭建Koa2 Server
  • 设计模式之模板方法模式
  • hibernate 继承映射
  • Introduction | Elasticsearch权威指南(中文版)
  • 【399天】跃迁之路——程序员高效学习方法论探索系列(实验阶段156-2018.03.11)...
  • 【跃迁之路】【444天】程序员高效学习方法论探索系列(实验阶段201-2018.04.25)...
  • EOS是什么
  • JavaScript HTML DOM
  • JavaScript 无符号位移运算符 三个大于号 的使用方法
  • JavaScript学习总结——原型
  • Perseus-BERT——业内性能极致优化的BERT训练方案
  • ubuntu 下nginx安装 并支持https协议
  • 动态规划入门(以爬楼梯为例)
  • 快速构建spring-cloud+sleuth+rabbit+ zipkin+es+kibana+grafana日志跟踪平台
  • 通过获取异步加载JS文件进度实现一个canvas环形loading图
  • 小程序上传图片到七牛云(支持多张上传,预览,删除)
  • 看到一个关于网页设计的文章分享过来!大家看看!
  • 整理一些计算机基础知识!
  • ​LeetCode解法汇总1276. 不浪费原料的汉堡制作方案
  • ​业务双活的数据切换思路设计(下)
  • #【QT 5 调试软件后,发布相关:软件生成exe文件 + 文件打包】
  • #我与Java虚拟机的故事#连载11: JVM学习之路
  • (1)bark-ml
  • (Java)【深基9.例1】选举学生会
  • (Redis使用系列) Springboot 使用redis实现接口Api限流 十
  • (附源码)计算机毕业设计SSM教师教学质量评价系统
  • (转)关于pipe()的详细解析
  • (转)利用ant在Mac 下自动化打包签名Android程序
  • (转)总结使用Unity 3D优化游戏运行性能的经验
  • **PyTorch月学习计划 - 第一周;第6-7天: 自动梯度(Autograd)**
  • .Net IE10 _doPostBack 未定义
  • .NET MVC之AOP
  • .Net 中的反射(动态创建类型实例) - Part.4(转自http://www.tracefact.net/CLR-and-Framework/Reflection-Part4.aspx)...
  • .NET文档生成工具ADB使用图文教程
  • [ 环境搭建篇 ] 安装 java 环境并配置环境变量(附 JDK1.8 安装包)
  • [1181]linux两台服务器之间传输文件和文件夹
  • [2016.7 day.5] T2
  • [BZOJ]4817: [Sdoi2017]树点涂色
  • [C/C++]关于C++11中的std::move和std::forward
  • [C++ 从入门到精通] 12.重载运算符、赋值运算符重载、析构函数
  • [Dxperience.8.*]报表预览控件PrintControl设置
  • [IE编程] 打开/关闭IE8的光标浏览模式(Caret Browsing)
  • [Java] 模拟Jdk 以及 CGLib 代理原理
  • [JavaWeb学习] idea新建web项目