• 单源点最短路径的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 }

    输出:


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

  • 相关阅读:
    宏定义函数 字符串 多行书写采用换行
    new / delete && new[] / delete[]
    删除vector中的重复元素
    AES加密算法通用版本
    字符串匹配KMP算法详解
    超外差【整理】
    LTE PDCCH 盲检测
    PDCCH学习
    PDCCH format 与传输模式之间的关系
    LTE测量事件主要有下面几种:Event A1、Event A2、Event A3、Event A4、Event A5、Event B1、Event B2
  • 原文地址:https://www.cnblogs.com/wxdjss/p/5515350.html
Copyright © 2020-2023  润新知