单源点最短路径的Dijkstra算法

在带权图(网)里,点A到点B所有路径中边的权值之和为最短的那一条路径,称为A,B两点之间的最短路径;并称路径上的第一个顶点为源点(Source),最后一个顶点为终点(Destination)。在无权图中,最短路径则是两点之间经历的边数最少的路径。实际上,只要把无权图上的每条边都看成是权值为1的边,那么无权图和带权图的最短路径是一致的。
   给定一个带权有向图G=(V,E),指定图G中的某一个顶点的V为源点,求出从V到其他各顶点之间的最短路径,这个问题称为单源点最短路径问题。
   迪杰斯特拉(Dijkstra)根据若按长度递增的次序生成从源点v0到其它顶点的最短路径,则当前正在生成的最短路径上除终点外,其余顶点的最短路径均已生成的这一思想,提出了按路径长度递增的次序产生最短路径的算法(在此,路径长度为路径上边和弧的权值之和)。Dijkstra算法的思想是:对带权有向图G=(V,E),设置两个顶点集合S和T=V-S;凡是以v0为源点并已确定了最短路径的终点(顶点)都并入到集合S,集合S的初态只含有源点v0;而未确定其最短路径的顶点均属于集合T,初态时集合T包含除了源点v0之外的其他顶点。按照各顶点与v0间最短路径的长度递增的次序,逐个把集合T中的各顶点的路径长度。并且,集合S中每加入一个新的顶点u,都要修改源点v0到集合T中剩余顶点的最短路径长度;也即,集合T中各顶点v新的最短路径长度值或是原来最短路径长度值,或是顶点u的最短路径长度值再加上顶点u到顶点v的路径长度值之和这两者中的较小值。这种把集合T中的顶点加入到集合S中的过程不断重复,直到集合T的顶点全部加入到集合S中为止。
   Dijkstra算法的实现中,以二维数组gm作为n个顶点带权有向图G=(V,E)的存储结构,并设置一个一维数组s(下标是0~n-1)用来标记集合S中已找到最短路径的顶点,而且规定:如果s[i]为0,则表示未找到源点v0到顶点vi的最短路径,也即此时vi在集合T中;如果s[i]为1,则表示已找到源点v0到顶点vi的最短路径(此时vi在集合S中).除了数组s外,还设置了一个数组dist(下标是0~n-1),用来保存从源点v0到终点vi的当前最短路径的长度.dist的初值为<v0,vi>边上的权值;若v0到vi没有边,则权值为&(无穷)。此后每当有一个新的顶点进入集合S中时,dist[i]值可能被修改变小.一维数组path(下标是0~n-1)用于保存最短路径长度中路径上边所经过的顶点序列;其中,path[i]保存从源点v0到终点vi当前最短路径中前一个顶点编号,它的初值是:如果v0到vi有边则置path[i]为v0的编号;如果v0到vi没有边则置path[i]为-1.
 参考代码:
 
 1 #include<stdio.h>
 2  #define MAXSIZE 6
 3  #define INF  32767
 4 
 5  void Ppath(int path[],int i,int v0)//先序递归查找最短路径(源点为v0)上的顶点
 6  {
 7    int k;
 8    k=path[i];
 9     if(k!=v0)//顶点Vk不是源点V0时
10    {
11      Ppath(path,k,v0);//递归查找顶点Vk的前一个顶点
12      printf("%d,",k);//输出顶点Vk
13     }
14   }
15 
16  void Dispath(int dist[],int path[],int s[],int v0,int n)//输出最短路径
17  {
18    int i;
19    for(i=0;i<n;i++)
20    if(s[i]==1)//顶点Vi在集合S中
21    {
22     printf("从%d到%d的最短路径长度为:%d,路径为:",v0,i,dist[i]);
23     printf("%d,",v0);//输出路径上的源点v0;
24     Ppath(path,i,v0);//输出路径上的中间顶点vi
25     printf("%d
",i);//输出路径上的终点
26      }
27    else
28   printf("从%d到%d不存在路径
",v0,i);
29 }
30 
31 void Dijkstra(int gm[][MAXSIZE],int v0,int n)//Dijkstra算法
32 {
33   int dist[MAXSIZE],path[MAXSIZE],s[MAXSIZE];
34   int i,j,k,mindis;
35   for(i=0;i<n;i++)
36   {
37    dist[i]=gm[v0][i];//v0到vi的最短路径初值赋给dist[i]
38    s[i]=0;//s[i]=0表示顶点vi属于T集
39    if(gm[v0][i]<INF)//路径初始化,INF为可取的最大常数
40     path[i]=v0;
41    else
42     path[i]=-1;//v0到vi没有边
43 }
44 s[v0]=1;path[v0]=0;//V0并入集合S且V0当前最短路径中无前一个顶点
45 for(i=0;i<n;i++)//对除V0外的n-1个顶点寻找最短路径,即循环n-1次
46  {
47   mindis=INF;
48   for(j=0;j<n;j++)//从当前集合T中选择一个路径长度最短的顶点Vk
49   if(s[j]==0&&dist[j]<mindis)
50   {
51    k=j;
52   mindis=dist[j];
53   }
54  s[k]=1;//顶点Vk加入集合S中
55   for(j=0;j<n;j++)//调整源点v0到集合T中任一顶点Vj的路径长度
56    if(s[j]==0)//顶点vj在集合T中
57     if(gm[k][j]<INF&&dist[k]+gm[k][j]<dist[j])//当V0到Vj的路径长度小于V0到Vk和Vk到Vj的路径长度时
58     {
59      dist[j]=dist[k]+gm[k][j];
60      path[j]=k;//Vk是当前最短路径中Vj的前一个顶点
61      }
62   }
63  Dispath(dist,path,s,v0,n);//输出最短路径
64 }
65 
66 void main()
67 {
68  int g[MAXSIZE][MAXSIZE]={{INF,20,15,INF,INF,INF},{2,INF,INF,INF,10,30},{INF,4,INF,INF,INF,10},
69                                           {INF,INF,INF,INF,INF,INF},{INF,INF,INF,15,INF,INF},{INF,INF,INF,4,10,INF}};//定义邻接矩阵g
70 Dijkstra(g,0,6);//求顶点0的最短路径
71 }

输出:


带权有向图及邻接矩阵示意:

原文地址:https://www.cnblogs.com/wxdjss/p/5515350.html