Dijkstra算法

2、Dijkstra算法介绍

  • 算法特点:

    迪科斯彻算法使用了广度优先搜索解决赋权有向图或者无向图的单源最短路径问题,算法最终得到一个最短路径树。该算法常用于路由算法或者作为其他图算法的一个子模块。

  • 算法的思路

    Dijkstra算法采用的是一种贪心的策略,声明一个数组dis来保存源点到各个顶点的最短距离和一个保存已经找到了最短路径的顶点的集合:T,初始时,原点 s 的路径权重被赋为 0 (dis[s] = 0)。若对于顶点 s 存在能直接到达的边(s,m),则把dis[m]设为w(s, m),同时把所有其他(s不能直接到达的)顶点的路径长度设为无穷大。初始时,集合T只有顶点s。 
    然后,从dis数组选择最小值,则该值就是源点s到该值对应的顶点的最短路径,并且把该点加入到T中,OK,此时完成一个顶点, 
    然后,我们需要看看新加入的顶点是否可以到达其他顶点并且看看通过该顶点到达其他点的路径长度是否比源点直接到达短,如果是,那么就替换这些顶点在dis中的值。 
    然后,又从dis中找出最小值,重复上述动作,直到T中包含了图的所有顶点。

3、Dijkstra算法示例演示

下面我求下图,从顶点v1到其他各个顶点的最短路径

这里写图片描述

首先第一步,我们先声明一个dis数组,该数组初始化的值为: 
这里写图片描述

我们的顶点集T的初始化为:T={v1}

既然是求 v1顶点到其余各个顶点的最短路程,那就先找一个离 1 号顶点最近的顶点。通过数组 dis 可知当前离v1顶点最近是 v3顶点。当选择了 2 号顶点后,dis[2](下标从0开始)的值就已经从“估计值”变为了“确定值”,即 v1顶点到 v3顶点的最短路程就是当前 dis[2]值。将V3加入到T中。 
为什么呢?因为目前离 v1顶点最近的是 v3顶点,并且这个图所有的边都是正数,那么肯定不可能通过第三个顶点中转,使得 v1顶点到 v3顶点的路程进一步缩短了。因为 v1顶点到其它顶点的路程肯定没有 v1到 v3顶点短.

OK,既然确定了一个顶点的最短路径,下面我们就要根据这个新入的顶点V3会有出度,发现以v3 为弧尾的有: < v3,v4 >,那么我们看看路径:v1–v3–v4的长度是否比v1–v4短,其实这个已经是很明显的了,因为dis[3]代表的就是v1–v4的长度为无穷大,而v1–v3–v4的长度为:10+50=60,所以更新dis[3]的值,得到如下结果: 
这里写图片描述

因此 dis[3]要更新为 60。这个过程有个专业术语叫做“松弛”。即 v1顶点到 v4顶点的路程即 dis[3],通过 < v3,v4> 这条边松弛成功。这便是 Dijkstra 算法的主要思想:通过“边”来松弛v1顶点到其余各个顶点的路程。

然后,我们又从除dis[2]和dis[0]外的其他值中寻找最小值,发现dis[4]的值最小,通过之前是解释的原理,可以知道v1到v5的最短距离就是dis[4]的值,然后,我们把v5加入到集合T中,然后,考虑v5的出度是否会影响我们的数组dis的值,v5有两条出度:< v5,v4>和 < v5,v6>,然后我们发现:v1–v5–v4的长度为:50,而dis[3]的值为60,所以我们要更新dis[3]的值.另外,v1-v5-v6的长度为:90,而dis[5]为100,所以我们需要更新dis[5]的值。更新后的dis数组如下图: 
这里写图片描述

然后,继续从dis中选择未确定的顶点的值中选择一个最小的值,发现dis[3]的值是最小的,所以把v4加入到集合T中,此时集合T={v1,v3,v5,v4},然后,考虑v4的出度是否会影响我们的数组dis的值,v4有一条出度:< v4,v6>,然后我们发现:v1–v5–v4–v6的长度为:60,而dis[5]的值为90,所以我们要更新dis[5]的值,更新后的dis数组如下图: 
这里写图片描述

然后,我们使用同样原理,分别确定了v6和v2的最短路径,最后dis的数组的值如下: 
这里写图片描述

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
public  class  Dijkstra {
 
     private  static  int  N =  1000 ;
     private  static  int [][] Graph = {
             0 1 5 , N, N, N, N, N, N },
             1 0 3 7 5 , N, N, N, N },
             5 3 0 , N,  1 7 , N, N, N },
             { N,  7 , N,  0 2 , N,  3 , N, N },
             { N,  5 1 2 0 3 6 9 , N },
             { N, N,  7 , N,  3 0 , N,  5 , N },
             { N, N, N,  3 6 , N,  0 2 7  },
             { N, N, N, N,  9 5 2 0 4  },
             { N, N, N, N, N, N,  7 4 0  } };
 
     public  static  void  main(String[] args) {
         dijkstra( 0 , Graph);
     }
 
     /**
      * Dijkstra最短路径。
      * 即图中"节点vs"到其它各个节点的最短路径。
      * @param vs 起始节点
      * @param Graph 图
      */
     public  static  void  dijkstra( int  vs,  int [][] Graph) {
         int  NUM = Graph[ 0 ].length;
         // 前驱节点数组
         int [] prenode =  new  int [NUM];
         // 最短距离数组
         int [] mindist =  new  int [NUM];
         // 该节点是否已经找到最短路径
         boolean [] find =  new  boolean [NUM];
         
         int  vnear =  0 ;
         
         for  ( int  i =  0 ; i < mindist.length; i++) {
             prenode[i] = i;
             mindist[i] = Graph[vs][i];
             find[i] =  false ;
         }
 
         find[vs] =  true ;
 
         for  ( int  v =  1 ; v < Graph.length; v++) {
 
             // 每次循环求得距离vs最近的节点vnear和最短距离min
             int  min = N;
             for  ( int  j =  0 ; j < Graph.length; j++) {
                 if  (!find[j] && mindist[j] < min) {
                     min = mindist[j];
                     vnear = j;
                 }
             }
             find[vnear] =  true ;
 
             // 根据vnear修正vs到其他所有节点的前驱节点及距离
             for  ( int  k =  0 ; k < Graph.length; k++) {
                 if  (!find[k] && (min + Graph[vnear][k]) < mindist[k]) {
                     prenode[k] = vnear;
                     mindist[k] = min + Graph[vnear][k];
                 }
             }
         }
         
         for  ( int  i =  0 ; i < NUM; i++) {
             System.out.println( "v"  + vs +  "...v"  + prenode[i] +  "->v"  + i +  ", s="  + mindist[i]);
         }
     }

转载于:https://www.cnblogs.com/Syiren/p/8776965.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值