`
hereson
  • 浏览: 1450066 次
  • 性别: Icon_minigender_1
  • 来自: 苏州
社区版块
存档分类
最新评论

单源最短路径算法---Dijkstra算法1

阅读更多
算法介绍
Dijkstra算法是由荷兰计算机科学家艾兹格·迪科斯彻发现的。算法解决的是有向图中最短路径问题。

举例来说,如果图中的顶点表示城市,而边上的权重表示著城市间开车行经的距离。 Dijkstra算法可以用来找到两个城市之间的最短路径。

Dijkstra算法的输入包含了一个有权重的有向图G,以及G中的一个来源顶点S。我们以V表示G中所有顶点的集合。每一个图中的边,都是两个顶点所形成的有序元素对。(u,v)表示从顶点u到v有路径相连。我们以E所有边的集合,而边的权重则由权重函数w: E → [0, ∞]定义。因此,w(u,v)就是从顶点u到顶点v的非负花费值(cost)。边的花费可以想像成两个顶点之间的距离。任两点间路径的花费值,就是该路径上所有边的花费值总和。已知有V中有顶点s及t,Dijkstra算法可以找到s到t的最低花费路径(i.e. 最短路径)。这个算法也可以在一个图中,找到从一个顶点s到任何其他顶点的最短路径。

算法描述
这个算法是通过为每个顶点v保留目前为止所找到的从s到v的最短路径来工作的。初始时,源点s的路径长度值被赋为0(d[s]=0),同时把所有其他顶点的路径长度设为无穷大,即表示我们不知道任何通向这些顶点的路径(对于V中所有顶点v除s外d[v]= ∞)。当算法结束时,d[v]中储存的便是从s到v的最短路径,或者如果路径不存在的话是无穷大。 Dijstra算法的基础操作是边的拓展:如果存在一条从u到v的边,那么从s到u的最短路径可以通过将边(u,v)添加到尾部来拓展一条从s到v的路径。这条路径的长度是d[u]+w(u,v)。如果这个值比目前已知的d[v]的值要小,我们可以用新值来替代当前d[v]中的值。拓展边的操作一直执行到所有的d[v]都代表从s到v最短路径的花费。这个算法经过组织因而当d[u]达到它最终的值的时候没条边(u,v)都只被拓展一次。

算法维护两个顶点集S和Q。集合S保留了我们已知的所有d[v]的值已经是最短路径的值顶点,而集合Q则保留其他所有顶点。集合S初始状态为空,而后每一步都有一个顶点从Q移动到S。这个被选择的顶点是Q中拥有最小的d[u]值的顶点。当一个顶点u从Q中转移到了S中,算法对每条外接边(u,v)进行拓展。

伪码
在下面的算法中,u:=Extract_Min(Q)在在顶点集Q中搜索有最小的d[u]值的顶点u。这个顶点被从集合Q中删除并返回给用户。

1 function Dijkstra(G, w, s)
2     for each vertex v in V[G]                        // 初始化
3           d[v] := infinity
4           previous[v] := undefined
5     d[s] := 0
6     S := empty set
7     Q := set of all vertices
8     while Q is not an empty set                      // Dijstra算法主体
9           u := Extract_Min(Q)
10           S := S union {u}
11           for each edge (u,v) outgoing from u
12                  if d[v] > d[u] + w(u,v)             // 拓展边(u,v)
13                        d[v] := d[u] + w(u,v)
14                        previous[v] := u

如果我们只对在s和t之间寻找一条最短路径的话,我们可以在第9行添加条件如果满足u=t的话终止程序。

现在我们可以通过迭代来回溯出s到t的最短路径

1 S := empty sequence
2 u := t
3 while defined u                                       
4       insert u to the beginning of S
5       u := previous[u]

现在序列S就是从s到t的最短路径的顶点集.

时间复杂度
我们可以用大O符号将Dijkstra算法的运行时间表示为边数m和顶点数n的函数。

Dijkstra算法最简单的实现方法是用一个链表或者数组来存储所有顶点的集合Q,所以搜索Q中最小元素的运算(Extract-Min(Q))只需要线性搜索Q中的所有元素。这样的话算法的运行时间是O(n2)。

对于边数少于n2稀疏图来说,我们可以用邻接表来更有效的实现Dijkstra算法。同时需要将一个二叉堆或者斐波纳契堆用作优先队列来寻找最小的顶点(Extract-Min)。当用到二叉堆的时候,算法所需的时间为O((m+n)log n),斐波纳契堆能稍微提高一些性能,让算法运行时间达到O(m + n log n)。

相关问题和算法
在Dijkstra算法的基础上作一些改动,可以扩展其功能。例如,有时希望在求得最短路径的基础上再列出一些次短的路径。为此,可先在原图上计算出最短路径,然后从图中删去该路径中的某一条边,在余下的子图中重新计算最短路径。对于原最短路径中的每一条边,均可求得一条删去该边后子图的最短路径,这些路径经排序后即为原图的一系列次短路径。

OSPF(open shortest path first, 开放最短路径优先)算法是Dijkstra算法在网络路由中的一个具体实现。

与Dijkstra算法不同,Bellman-Ford算法可用于具有负花费边的图,只要图中不存在总花费为负值且从源点 s 可达的环路(如果有这样的环路,则最短路径不存在,因为沿环路循环多次即可无限制的降低总花费)。

与最短路径问题有关的一个问题是旅行商问题(traveling salesman problem),它要求找出通过所有顶点恰好一次且最终回到源点的最短路径。该问题是NP难的;换言之,与最短路径问题不同,旅行商问题不太可能具有多项式时间算法。

如果有已知信息可用来估计某一点到目标点的距离,则可改用A*算法 href="http://www.roboticfan.com/college/knowledge/200606/146.shtml">A*算法,以减小最短路径的搜索范围。

参考
E. W. Dijkstra: A note on two problems in connexion with graphs. In: Numerische Mathematik. 1 (1959), S. 269–271
Thomas H. Cormen, Charles E. Leiserson, Ronald L. Rivest, and Clifford Stein. Introduction to Algorithms, Second Edition. MIT Press and McGraw-Hill, 2001. ISBN 0262032937. Section 24.3: Dijkstra's algorithm, pp.595–601.
分享到:
评论
1 楼 Emy 2008-05-01  
很有价值 

相关推荐

    用Dijkstra算法求图中单源最短路径

    "用Dijkstra算法求图中单源最短路径" Dijkstra算法是一种常用的图搜索算法,用于解决单源最短路径问题。单源最短路径问题是指从一个出发顶点到所有可达顶点的最短路径问题。Dijkstra算法的原理是通过将图中的每个...

    单源最短路径实验报告

    在【描述】中提到的实验报告是针对计算机科学与技术专业学生的,旨在通过实践加深对单源最短路径算法的理解。实验中特别提到了贪心算法,这是一种解决问题的策略,它在每一步选择局部最优解,希望通过一系列局部最优...

    贪心算法 Dijkstra 单源最短路径

    用C++实现的贪心算法 Dijkstra 单源最短路径,并包含大量的注释,对理解程序很有帮助

    贪心算法之应用-单源最短路径-Dijkstra算法学习

    贪心算法之应用-单源最短路径-Dijkstra算法学习

    基于C++进行单源最短路径算法(SSSP)的探究【100012197】

    总之,基于C++的单源最短路径算法是图论中的重要工具,通过理解图的结构、选择合适的数据结构和算法,我们可以有效地解决实际问题。"delta-stepping"算法提供了一种平衡时间和空间复杂度的解决方案,对于教学和实际...

    dijkstra最短路径算法--算法论文

    Dijkstra最短路径算法 Dijkstra算法是图论中应用最广的算法之一,广泛应用于解决最短路径问题。在该算法中,我们可以将问题转换为图论问题,然后使用Dijkstra算法来求解。该算法的基本思路是按照源点s到其他各个...

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

    ### 单源最短路径问题的Dijkstra算法详解 #### 一、算法背景与应用场景 在图论中,单源最短路径问题是指在一个带有非负权重边的有向图或无向图中找到从一个指定顶点(称为源点)到其他所有顶点的最短路径的问题。...

    30-图算法-单源最短路径-Dijkstra1

    【正文】 ...总结,Dijkstra算法是一种高效的解决单源最短路径问题的方法,广泛应用于路由选择、网络优化等领域。尽管在处理大规模负权重图时可能遇到挑战,但在许多实际应用中,它仍然是首选的解决方案。

    最短路径算法--vb源码

    - **Dijkstra算法**:一种广泛使用的单源最短路径算法,适用于加权无环图。它通过维护一个优先队列来逐步更新所有顶点的最短路径。 - **Bellman-Ford算法**:能处理含有负权边的图,但时间复杂度比Dijkstra高,为O...

    单源最短路径(算法 代码)

    本篇文章将深入探讨单源最短路径算法,并通过VC++环境下的C++代码实现来帮助理解。 首先,我们要了解几种常见的单源最短路径算法: 1. **Dijkstra算法**:由荷兰计算机科学家艾兹格·迪科斯彻于1956年提出,适用于...

    图论算法库 C++ 语言实现 代码内容 图论算法库,包括以下算法: 单源最短路径 Dijkstra 算法 单源最短路径 Bell

    图论算法库 C++ 语言实现 代码内容 图论算法库,包括以下算法: 单源最短路径 Dijkstra 算法 单源最短路径 Bellman-Ford 算法 最小生成树 Prim 算法 每对节点间最短路径 Flod-Warshall 算法

    单源最短路径dijkstra模板

    1. Dijkstra算法:一种常用的图算法,用于解决单源最短路径问题。 2. 邻接矩阵:一种图存储方式,使用矩阵来存储图的边信息。 3. 距离数组:用于存储从源节点到其他所有节点的最短距离。 4. 前驱节点:用于存储每个...

    最短路径算法Dijkstra源代码

    最短路径算法Dijkstra是图论中的一个经典算法,由荷兰计算机科学家艾兹格·迪科斯彻在1956年提出。该算法主要用于寻找带权重的有向或无向图中,从一个特定顶点到其他所有顶点的最短路径。Dijkstra算法的核心思想是...

    python实现有向图单源最短路径迪杰斯特拉 算法

    迪杰斯特拉(Dijkstra)算法是一种用于寻找有向图中单源最短路径的经典算法,由荷兰计算机科学家艾兹格·迪杰斯特拉在1956年提出。在这个算法中,我们从一个源节点开始,逐步扩展最短路径到其他节点,直到到达所有...

    用Dijkstra算法实现单源最短路径问题

    ### Dijkstra算法实现单源最短路径问题 #### 算法原理 Dijkstra算法是一种用于寻找图中两点之间的最短路径的经典算法。它适用于所有边权重非负的情况,并且可以高效地解决单源最短路径问题。在该问题中,我们需要...

    单源最短路径 直接copy 运行即可,里面有测试结果

    ### 单源最短路径算法解析与应用实例 在图论和网络理论中,寻找从一个顶点到图中其他所有顶点的最短路径是一个常见的问题,这被称为单源最短路径问题。该问题在各种领域都有广泛的应用,如交通规划、网络路由选择、...

    基于贪心法求解单源最短路径问题.docx

    实验目的:理解贪心法的核心思想、贪心法的求解过程,并从算法分析与设计的角度,对单源最短路径问题的 Dijkstra 算法求解有进一步理解。 实验内容: 1. 问题描述:给定一个有向带权图 G=(V,E),其中每条边的权是...

    Dijkstra单源最短路径代码 C/C++实现

    DIJKSTRA单源最短路径算法C/C++实现,内有注释,输入邻接矩阵,输入源点到终点最短路径长度。

    Dijkstra最短路径算法

    **Dijkstra最短路径算法**是图论中的一个重要算法,由荷兰计算机科学家艾兹格·迪科斯彻在1956年提出,主要用于解决单源最短路径问题。这个算法能够找到图中一个起点到其他所有顶点的最短路径,尤其适用于有向图或无...

Global site tag (gtag.js) - Google Analytics