-
迪杰斯特拉算法
编辑
-
中文名
- 迪克斯特拉算法 外文名
- Dijkstra's Algorithm
-
分 类
- 计算机算法 用 途
- 单源最短路径问题
目录
Dijkstra算法是典型的
算法。
Dijkstra算法是很有代表性的
算法。Dijkstra一般的表述通常有两种方式,一种用永久和临时标号方式,一种是用
OPEN, CLOSE表的方式,这里均采用永久和临时标号的方式。注意该算法要求图中不存在负权边。
[2]
1.首先,引入一个辅助向量D,它的每个分量 D
表示当前所找到的
从起始点
(即源点
)到其它每个顶点
的长度。
![](https://i-blog.csdnimg.cn/blog_migrate/ee97014a38fbe2d5b732042f7fd5a3ae.png)
![](https://i-blog.csdnimg.cn/blog_migrate/6974ded72c840d4ac670f57ada8bc835.png)
![](https://i-blog.csdnimg.cn/blog_migrate/6974ded72c840d4ac670f57ada8bc835.png)
![](https://i-blog.csdnimg.cn/blog_migrate/218fb1cc8dec27a1854e4bc0e1d4ad10.png)
2.D的初始状态为:若从
到
有弧(即从
到
存在连接边),则D
为弧上的权值(即为从
到
的边的权值);否则置D
为∞。
![](https://i-blog.csdnimg.cn/blog_migrate/6974ded72c840d4ac670f57ada8bc835.png)
![](https://i-blog.csdnimg.cn/blog_migrate/218fb1cc8dec27a1854e4bc0e1d4ad10.png)
![](https://i-blog.csdnimg.cn/blog_migrate/6974ded72c840d4ac670f57ada8bc835.png)
![](https://i-blog.csdnimg.cn/blog_migrate/218fb1cc8dec27a1854e4bc0e1d4ad10.png)
![](https://i-blog.csdnimg.cn/blog_migrate/ee97014a38fbe2d5b732042f7fd5a3ae.png)
![](https://i-blog.csdnimg.cn/blog_migrate/6974ded72c840d4ac670f57ada8bc835.png)
![](https://i-blog.csdnimg.cn/blog_migrate/218fb1cc8dec27a1854e4bc0e1d4ad10.png)
![](https://i-blog.csdnimg.cn/blog_migrate/ee97014a38fbe2d5b732042f7fd5a3ae.png)
显然,长度为 D
= Min{ D |
∈V } 的路径就是从
出发到顶点
的长度最短的一条路径,此路径为(
)。
![](https://i-blog.csdnimg.cn/blog_migrate/0749be93b0bc176b937b591204d4a27c.png)
![](https://i-blog.csdnimg.cn/blog_migrate/218fb1cc8dec27a1854e4bc0e1d4ad10.png)
![](https://i-blog.csdnimg.cn/blog_migrate/6974ded72c840d4ac670f57ada8bc835.png)
![](https://i-blog.csdnimg.cn/blog_migrate/9cd7bb597f950ae30308e775cb068f36.png)
![](https://i-blog.csdnimg.cn/blog_migrate/16403fce3150543cad8e7b02be895526.png)
3.那么,下一条长度次短的是哪一条呢?也就是找到从源点
到下一个顶点的最短路径长度所对应的顶点,且这条最短路径长度仅次于从源点
到顶点
的最短路径长度。
![](https://i-blog.csdnimg.cn/blog_migrate/6974ded72c840d4ac670f57ada8bc835.png)
![](https://i-blog.csdnimg.cn/blog_migrate/6974ded72c840d4ac670f57ada8bc835.png)
![](https://i-blog.csdnimg.cn/blog_migrate/9cd7bb597f950ae30308e775cb068f36.png)
假设该次短路径的终点是
,则可想而知,这条路径要么是(
),或者是(
)。它的长度或者是从
到
的弧上的权值,或者是D
加上从
到
的弧上的权值。
![](https://i-blog.csdnimg.cn/blog_migrate/57d9503337e3a36d86b4f6ff62c2a82f.png)
![](https://i-blog.csdnimg.cn/blog_migrate/a446e504782bee6bbc6acc377827b8ab.png)
![](https://i-blog.csdnimg.cn/blog_migrate/adfd5adc06d72d4461bb6a828f94f4a9.png)
![](https://i-blog.csdnimg.cn/blog_migrate/6974ded72c840d4ac670f57ada8bc835.png)
![](https://i-blog.csdnimg.cn/blog_migrate/57d9503337e3a36d86b4f6ff62c2a82f.png)
![](https://i-blog.csdnimg.cn/blog_migrate/0749be93b0bc176b937b591204d4a27c.png)
![](https://i-blog.csdnimg.cn/blog_migrate/cb20b2c886a85ea68a7ae7bac1f0b95f.png)
![](https://i-blog.csdnimg.cn/blog_migrate/57d9503337e3a36d86b4f6ff62c2a82f.png)
4.一般情况下,假设S为已求得的从源点
出发的最短路径长度的顶点的集合,则可证明:下一条次最短路径(设其终点为
)要么是弧(
),或者是从源点
出发的中间只经过S中的顶点而最后到达顶点
的路径。
![](https://i-blog.csdnimg.cn/blog_migrate/6974ded72c840d4ac670f57ada8bc835.png)
![](https://i-blog.csdnimg.cn/blog_migrate/b16ba7064043a6234e2b47137ca3c82d.png)
![](https://i-blog.csdnimg.cn/blog_migrate/7da24e8adc88ae58c91a8e9df713eb9b.png)
![](https://i-blog.csdnimg.cn/blog_migrate/6974ded72c840d4ac670f57ada8bc835.png)
![](https://i-blog.csdnimg.cn/blog_migrate/b16ba7064043a6234e2b47137ca3c82d.png)
因此,下一条长度次短的的最短路径长度必是D
= Min{ D
|
∈V-S },其中D
要么是弧(
)上的权值,或者是D
(
∈S)和弧(
,
)上的权值之和。
![](https://i-blog.csdnimg.cn/blog_migrate/0749be93b0bc176b937b591204d4a27c.png)
![](https://i-blog.csdnimg.cn/blog_migrate/ee97014a38fbe2d5b732042f7fd5a3ae.png)
![](https://i-blog.csdnimg.cn/blog_migrate/218fb1cc8dec27a1854e4bc0e1d4ad10.png)
![](https://i-blog.csdnimg.cn/blog_migrate/ee97014a38fbe2d5b732042f7fd5a3ae.png)
![](https://i-blog.csdnimg.cn/blog_migrate/4efff6509e15ac17232d4b333326e44b.png)
![](https://i-blog.csdnimg.cn/blog_migrate/4fc4951c54e6932383f45e54d14cef25.png)
![](https://i-blog.csdnimg.cn/blog_migrate/57d9503337e3a36d86b4f6ff62c2a82f.png)
![](https://i-blog.csdnimg.cn/blog_migrate/57d9503337e3a36d86b4f6ff62c2a82f.png)
![](https://i-blog.csdnimg.cn/blog_migrate/218fb1cc8dec27a1854e4bc0e1d4ad10.png)
算法描述如下:
1)令arcs表示弧上的权值。若弧不存在,则置arcs为∞(在本程序中为MAXCOST)。S为已找到的从
出发的的终点的集合,初始状态为空集。那么,从
出发到图上其余各顶点
可能达到的长度的初值为D=arcs[Locate Vex(G,
)],
∈V;
![](https://i-blog.csdnimg.cn/blog_migrate/6974ded72c840d4ac670f57ada8bc835.png)
![](https://i-blog.csdnimg.cn/blog_migrate/6974ded72c840d4ac670f57ada8bc835.png)
![](https://i-blog.csdnimg.cn/blog_migrate/218fb1cc8dec27a1854e4bc0e1d4ad10.png)
![](https://i-blog.csdnimg.cn/blog_migrate/218fb1cc8dec27a1854e4bc0e1d4ad10.png)
![](https://i-blog.csdnimg.cn/blog_migrate/218fb1cc8dec27a1854e4bc0e1d4ad10.png)
2)选择
,使得D
=Min{ D |
∈V-S } ;
![](https://i-blog.csdnimg.cn/blog_migrate/9cd7bb597f950ae30308e775cb068f36.png)
![](https://i-blog.csdnimg.cn/blog_migrate/0749be93b0bc176b937b591204d4a27c.png)
![](https://i-blog.csdnimg.cn/blog_migrate/218fb1cc8dec27a1854e4bc0e1d4ad10.png)
按路径长度
递增次序产生算法:
把顶点集合V分成两组:
(1)S:已求出的顶点的集合(初始时只含有源点V0)
(2)V-S=T:尚未确定的顶点集合
将T中顶点按递增的次序加入到S中,保证:
(1)从源点V0到S中其他各顶点的长度都不大于从V0到T中任何顶点的最短路径长度
(2)每个顶点对应一个距离值
S中顶点:从V0到此顶点的长度
T中顶点:从V0到此顶点的只包括S中顶点作中间顶点的最短路径长度
依据:可以证明V0到T中顶点Vk的,或是从V0到Vk的直接路径的权值;或是从V0经S中顶点到Vk的路径权值之和
(
反证法可证)
求最短路径步骤
算法步骤如下:
G={V,E}
1. 初始时令 S={V0},T=V-S={其余顶点},T中顶点对应的距离值
若存在<V0,Vi>,d(V0,Vi)为<V0,Vi>弧上的权值
若不存在<V0,Vi>,d(V0,Vi)为∞
2. 从T中选取一个与S中顶点有关联边且权值最小的顶点W,加入到S中
3. 对其余T中顶点的距离值进行修改:若加进W作中间顶点,从V0到Vi的距离值缩短,则修改此距离值
重复上述步骤2、3,直到S中包含所有顶点,即W=Vi为止
pascal语言
下面是该算法的Pascal程序
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
69
70
71
|
type
bool=
array
[
1..10
]ofboolean;
arr=
array
[
0..10
]ofinteger;
var
a:
array
[
1..10
,
1..10
]ofinteger;
//存储图的邻接数组,无边为10000
c,d,e:arr;
//c为最短路径数值,d为各点前趋,
t:bool;
//e:路径,t为辅助数组
i,j,n,m:
integer
;
inf,outf:text;
procedureinit;
//不同题目邻接数组建立方式不一样
begin
assign(inf,inputfile);
assign(outf,outputfile);
reset(inf);
rewrite(outf);
read(inf,n);
fori:=1tondo
begin
forj:=1tondo
begin
read(inf,a[i,j]);
ifa[i,j]=0then
a[i,j]:=
10000
;
end
;
end
;
end
;
proceduredijkstra(qi:
integer
;t:bool;varc
{,d}
:arr);
//qi起点,{}中为求路径部分,不需求路径时可以不要
var
i,j,k,min:
integer
;
|