最短路径算法-迪杰斯特拉(Dijkstra)算法

迪杰斯特拉(Dijkstra)算法是典型最短路径算法,用于计算一个节点到其他节点的最短路径。
它的主要特点是以起始点为中心向外层层扩展(广度优先遍历思想),直到扩展到终点为止。

基本思想

  1. 通过Dijkstra计算图G中的最短路径时,需要指定一个起点D(即从顶点D开始计算)。
  2. 此外,引进两个数组S和U。S的作用是记录已求出最短路径的顶点(以及相应的最短路径长度),而U则是记录还未求出最短路径的顶点(以及该顶点到起点D的距离)。
  3. 初始时,数组S中只有起点D;数组U中是除起点D之外的顶点,并且数组U中记录各顶点到起点D的距离。如果顶点与起点D不相邻,距离为无穷大。
  4. 然后,从数组U中找出路径最短的顶点K,并将其加入到数组S中;同时,从数组U中移除顶点K。接着,更新数组U中的各顶点到起点D的距离。
  5. 重复第4步操作,直到遍历完所有顶点。

迪杰斯特拉(Dijkstra)算法图解

以上图为例,来对迪杰斯特拉进行算法演示(以顶点D为起点)。

初始状态:S是已计算出最短路径的顶点集合,U是未计算除最短路径的顶点的集合!
第1步:将顶点D加入到S中。
    此时,S={D(0)}, U={A(∞),B(∞),C(3),E(4),F(∞),G(∞)}。     注:C(3)表示C到起点D的距离是3。

第2步:将顶点C加入到S中。
    上一步操作之后,U中顶点C到起点D的距离最短;因此,将C加入到S中,同时更新U中顶点的距离。以顶点F为例,之前F到D的距离为∞;但是将C加入到S之后,F到D的距离为9=(F,C)+(C,D)。
    此时,S={D(0),C(3)}, U={A(∞),B(23),E(4),F(9),G(∞)}。

第3步:将顶点E加入到S中。
    上一步操作之后,U中顶点E到起点D的距离最短;因此,将E加入到S中,同时更新U中顶点的距离。还是以顶点F为例,之前F到D的距离为9;但是将E加入到S之后,F到D的距离为6=(F,E)+(E,D)。
    此时,S={D(0),C(3),E(4)}, U={A(∞),B(23),F(6),G(12)}。

第4步:将顶点F加入到S中。
    此时,S={D(0),C(3),E(4),F(6)}, U={A(22),B(13),G(12)}。

第5步:将顶点G加入到S中。
    此时,S={D(0),C(3),E(4),F(6),G(12)}, U={A(22),B(13)}。

第6步:将顶点B加入到S中。
    此时,S={D(0),C(3),E(4),F(6),G(12),B(13)}, U={A(22)}。

第7步:将顶点A加入到S中。
    此时,S={D(0),C(3),E(4),F(6),G(12),B(13),A(22)}。

此时,起点D到各个顶点的最短距离就计算出来了:A(22) B(13) C(3) D(0) E(4) F(6) G(12)

迪杰斯特拉算法的C#实现

using System;/**
* Dijkstra算法获取最短路径
*/
public class Dijkstra
{public static int MAX = int.MaxValue;// 邻接表中表对应的链表的顶点public class ENode{public int ivex;       // 该边所指向的顶点的位置public int weight;     // 该边的权public ENode nextEdge; // 指向下一条弧的指针}// 邻接表中表的顶点public class VNode{public char data;          // 顶点信息public ENode firstEdge;    // 指向第一条依附该顶点的弧}// 边的结构体public class EData{public char start; // 边的起点public char end;   // 边的终点public int weight; // 边的权重public EData(char start, char end, int weight){this.start = start;this.end = end;this.weight = weight;}}public int mEdgNum;    // 边的数量public VNode[] mVexs;  // 顶点数组/** 创建图** 参数说明:*     vexs  -- 顶点数组*     edges -- 边*/public Dijkstra(char[] vexs, EData[] edges){// 初始化"顶点数"和"边数"int vlen = vexs.Length;int elen = edges.Length;// 初始化"顶点"mVexs = new VNode[vlen];for (int i = 0; i < mVexs.Length; i++){mVexs[i] = new VNode();mVexs[i].data = vexs[i];mVexs[i].firstEdge = null;}// 初始化"边"mEdgNum = elen;for (int i = 0; i < elen; i++){// 读取边的起始顶点和结束顶点char c1 = edges[i].start;char c2 = edges[i].end;int weight = edges[i].weight;// 读取边的起始顶点和结束顶点int p1 = GetPosition(c1);int p2 = GetPosition(c2);// 初始化node1ENode node1 = new ENode();node1.ivex = p2;node1.weight = weight;// 将node1链接到"p1所在链表的末尾"if (mVexs[p1].firstEdge == null)mVexs[p1].firstEdge = node1;elseLinkLast(mVexs[p1].firstEdge, node1);// 初始化node2ENode node2 = new ENode();node2.ivex = p1;node2.weight = weight;// 将node2链接到"p2所在链表的末尾"if (mVexs[p2].firstEdge == null)mVexs[p2].firstEdge = node2;elseLinkLast(mVexs[p2].firstEdge, node2);}}/** 将node节点链接到list的最后*/private void LinkLast(ENode list, ENode node){ENode p = list;while (p.nextEdge != null)p = p.nextEdge;p.nextEdge = node;}/** 返回ch位置*/private int GetPosition(char ch){for (int i = 0; i < mVexs.Length; i++)if (mVexs[i].data == ch)return i;return -1;}/** 获取边<start, end>的权值;若start和end不是连通的,则返回无穷大。*/private int GetWeight(int start, int end){if (start == end)return 0;ENode node = mVexs[start].firstEdge;while (node != null){if (end == node.ivex)return node.weight;node = node.nextEdge;}return MAX;}/** Dijkstra最短路径。* 即,统计图中"起点D"到其它各个顶点的最短路径。** 参数说明:*       vs -- 起始顶点(start vertex)。*     prev -- 前驱顶点数组。即,prev[i]的值是"起点D"到"顶点i"的最短路径所经历的全部顶点中,位于"顶点i"之前的那个顶点。*     dist -- 长度数组。即,dist[i]是"起点D"到"顶点i"的最短路径的长度。*/public void CalcDijkstra(int vs, int[] prev, int[] dist){// flag[i]=true表示"起点D"到"顶点i"的最短路径已成功获取。bool[] flag = new bool[mVexs.Length];// 初始化for (int i = 0; i < mVexs.Length; i++){flag[i] = false;            // 顶点i的最短路径还没获取到。prev[i] = 0;                // 顶点i的前驱顶点为0。dist[i] = GetWeight(vs, i); // 顶点i的最短路径为"起点D"到"顶点i"的权。}// 对"起点D"自身进行初始化flag[vs] = true;dist[vs] = 0;// 遍历mVexs.Length-1次;每次找出一个顶点的最短路径。int k = 0;for (int i = 1; i < mVexs.Length; i++){// 寻找当前最小的路径// 即,在未获取最短路径的顶点中,找到离起点D最近的顶点(k)。int min = MAX;for (int j = 0; j < mVexs.Length; j++){if (flag[j] == false && dist[j] < min){min = dist[j];k = j;}}// 标记"顶点k"为已经获取到最短路径flag[k] = true;// 更新当前最短路径和前驱顶点// 即,更新"未获取最短路径的顶点的最短路径和前驱顶点"。for (int j = 0; j < mVexs.Length; j++){int tmp = GetWeight(k, j);tmp = (tmp == MAX ? MAX : (min + tmp)); // 防止溢出if (flag[j] == false && (tmp < dist[j])){dist[j] = tmp;prev[j] = k;}}}// 打印dijkstra最短路径的结果Console.WriteLine("dijkstra({0}): \n", mVexs[vs].data);for (int i = 0; i < mVexs.Length; i++){Console.WriteLine("shortest({0}, {1})={2}\n", mVexs[vs].data, mVexs[i].data, dist[i]);}}public static void Main(string[] args){char[] vexs = { 'A', 'B', 'C', 'D', 'E', 'F', 'G' };EData[] edges = {// 起点 终点 权重new EData('A', 'B', 12),new EData('A', 'F', 16),new EData('A', 'G', 14),new EData('B', 'C', 10),new EData('B', 'F',  7),new EData('C', 'D',  3),new EData('C', 'E',  5),new EData('C', 'F',  6),new EData('D', 'E',  4),new EData('E', 'F',  2),new EData('E', 'G',  8),new EData('F', 'G',  9),};Dijkstra pG = new Dijkstra(vexs, edges);// dijkstra算法获取"起点D"到其它各个顶点的最短距离int[] prev = new int[pG.mVexs.Length];int[] dist = new int[pG.mVexs.Length];pG.CalcDijkstra(3, prev, dist);Console.ReadKey();}
}

执行结果

参考:https://www.cnblogs.com/skywang12345/p/3711514.html

最短路径算法-迪杰斯特拉(Dijkstra)算法相关推荐

  1. 数据结构与算法(7-4)最短路径(迪杰斯特拉(Dijkstra)算法、弗洛伊德(Floyd)算法)

    目录 一.最短路径概念 二.迪杰斯特拉(Dijkstra)算法(单源最短路径) 1.原理 2.过程 3.代码 三.弗洛伊德(Floyd)算法(多源最短路径) 1.原理 2.存储 3.遍历 4.代码 参 ...

  2. java数据结构和算法——迪杰斯特拉(Dijkstra)算法

    目录 一.迪杰斯特拉(Dijkstra)算法介绍 二.迪杰斯特拉(Dijkstra)算法过程 三.迪杰斯特拉(Dijkstra)算法--应用场景(最短路径问题) 四.迪杰斯特拉(Dijkstra)算法 ...

  3. 最短路径之迪杰斯特拉(Dijkstra)算法

    定义 迪杰斯特拉算法(Dijkstra)是由荷兰计算机科学家狄克斯特拉于1959 年提出的,因此又叫狄克斯特拉算法.是从一个顶点到其余各顶点的最短路径算法,解决的是有权图中最短路径问题.迪杰斯特拉算法 ...

  4. 最短路径:迪杰斯特拉(Dijkstra)算法图解

    Dijkstra算法可以求解带权图中最短路径. 算法思想:将所有节点分为两个区域,已知区域和未知区域.最开始的时候,将起点加入到已知区域,其他点加入未知区域,比较已知区域到未知区域的所有连线,最短的路 ...

  5. 059.迪杰斯特拉(Dijkstra)算法的原理以及解决最短路径问题

    1. 迪杰斯特拉(Dijkstra)算法的原理 1.1. 算法应用场景-最短路径问题 1.2. 基本介绍 1.3. 步骤详解 1.4. 思路解析 1.5. 图解步骤 2. 迪杰斯特拉(Dijkstra ...

  6. 迪杰斯特拉(Dijkstra)算法解决最短路径问题

    Dijkstra 算法介绍 迪杰斯特拉算法(Dijkstra)是由荷兰计算机科学家狄克斯特拉于1959年提出的,因此又叫狄克斯特拉算法.迪杰斯特拉(Dijkstra)算法是最经典的最短路径算法之一,用 ...

  7. Java迪杰斯特拉(Dijkstra)算法与弗洛伊德(Floyd)算法

    1.Java迪杰斯特拉(Dijkstra)算法与弗洛伊德(Floyd)算法 1.1 迪杰斯特拉(Dijkstra)算法 1.1.1 迪杰斯特拉(Dijkstra)算法介绍 迪杰斯特拉(Dijkstra ...

  8. java实现迪杰斯特拉(Dijkstra)算法求解最短路问题

    迪杰斯特拉(Dijkstra)算法是由荷兰计算机科学家狄克斯特拉于1959年提出的.是寻找从一个顶点到其余各顶点的最短路径算法,可用来解决最短路径问题. 迪杰斯特拉算法采用贪心算法的策略,将所有顶点分 ...

  9. 数据结构——图——迪杰斯特拉(Dijkstra )算法

    数据结构--图--迪杰斯特拉(Dijkstra )算法 这是一个按路径长度递增的次序产生最短路径的算法.它的思路大体是这样的. 比如说要求图7-7-3中顶点v0到顶点v1的最短距离,没有比这更简单的了 ...

最新文章

  1. ButterKnife与RxBinding中文件重复问题解决
  2. 4.3.6无分类编址CIDR(构成超网)
  3. Arthas实践--获取到Spring Context,然后为所欲为
  4. 百度打造AIR智能道路系统,可降低90%交通事故
  5. count(*),count(1),count(0)效率
  6. Ext.data.GroupingStore详解
  7. Js中String对象
  8. nvme驱动_用户态NVMe运维利器 SPDK NVMe 字符设备
  9. uni-app路由的封装
  10. Algs4-1.3.10中序表达式转为后序表达式(第二次实现)
  11. TCP模块如何处理数据包
  12. 杂项-数学软件:MATLAB
  13. Django常见面试题总结(一)
  14. java链接Kepserver报错误码[0x8001FFFF]
  15. python cad按范围裁剪
  16. 口袋妖怪金心银魂详细图文攻略(下)及游戏下载
  17. SQLyog 报错2058 :连接 mysql 8.0.11 解决方法
  18. python猜数游戏续_python猜数游戏续改编
  19. 英语--日常生活名词收集 Daily Life Nouns Collection
  20. 精简字体库:FontSubsetGUI,FontForge

热门文章

  1. BA无标度网络模型构造算法
  2. 如何把mkv转成mp4?
  3. 十几分钟学会搭建动态网站
  4. 2025年的技术:为第四次工业革命做准备
  5. 从一个角色建模师角度谈谈游戏角色设计
  6. SAP:FMALV转OOALV函数及多行多列实现方法
  7. HTML5基础知识笔记(更新)
  8. 实现智能读报(逐字朗读+自动滚屏)
  9. 小程序自定义组件使用及跨级传值
  10. 基于springboot+vue的小区物业管理系统