zoukankan      html  css  js  c++  java
  • 单源点最短路径的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 }

    输出:


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

  • 相关阅读:
    内存管理简介之Buddy算法和slab分配
    进程通信方式介绍
    Linux内核网络栈实现分析(十一)驱动程序层(下)
    Linux内核网络协议栈深入分析(二)sk_buff的操作函数
    Linux内核网络协议栈深入分析(一)与sk_buff有关的几个重要的数据结构
    内核源码学习:伙伴算法
    寒假Day16Dinic模板更新+优化
    寒假Day20:数位dp
    寒假Day21:Catalan Square卡特兰数 JAVA写大数
    寒假Day17UVALive3231Fair Share(最大流+二分)
  • 原文地址:https://www.cnblogs.com/wxdjss/p/5515350.html
Copyright © 2011-2022 走看看