通过根据其顶点的拓扑排序放宽加权DAG(有向无环图)G =(V, E)的边缘, 我们可以找出source(V + E)时间中来自单个源的最短路径。由于即使存在负权重边缘, 也不会存在负权重循环, 因此最短路径总是很好地描述。
DAG - SHORTEST - PATHS (G, w, s)
1. Topologically sort the vertices of G.
2. INITIALIZE - SINGLE- SOURCE (G, s)
3. for each vertex u taken in topologically sorted order
4. do for each vertex v ∈ Adj [u]
5. do RELAX (u, v, w)
该数据的运行时间由第1行和第3-5行的for循环确定。拓扑排序可以在∅(V + E)时间中实现。在第3-5行的for循环中, 就像Dijkstra的算法一样, 每个顶点有一个重复。对于每个顶点, 离开顶点的边都将被精确检查一次。与Dijkstra的算法不同, 我们每个边缘仅使用O(1)时间。因此, 运行时间为∅(V + E), 在图表的邻接表描绘的大小上是线性的。
例:
data:image/s3,"s3://crabby-images/03b83/03b8302f1c14359b6202cd0207a239bfea1a5e87" alt="有向无环图中的单源最短路径"
步骤1:要对顶点进行拓扑排序, 请应用DFS(深度优先搜索), 然后通过减小完成时间的顺序以线性顺序排列顶点。
data:image/s3,"s3://crabby-images/92928/9292817cdd86431baae76cdd6a08f416908d608c" alt="有向无环图中的单源最短路径"
data:image/s3,"s3://crabby-images/187ef/187efb94ac56ac6791cd8fc00eda843729a3071f" alt="有向无环图中的单源最短路径"
现在, 按照拓扑排序的顺序获取每个顶点, 并放松每个边缘。
data:image/s3,"s3://crabby-images/39b4c/39b4c538f3adf333b8b012859052fa17297f5d62" alt="有向无环图中的单源最短路径"
adj [s] →t, x
0 + 3 < ∞
d [t] ← 3
0 + 2 < ∞
d [x] ← 2
data:image/s3,"s3://crabby-images/8014a/8014a525b84d833b7ef46b578b6a64338fcfe882" alt="有向无环图中的单源最短路径"
adj [t] → r, x
3 + 1 < ∞
d [r] ← 4
3 + 5 ≤ 2
data:image/s3,"s3://crabby-images/e9809/e9809e97f314c23691cfc7feef9faa5297e82e43" alt="有向无环图中的单源最短路径"
adj [x] → y
2 - 3 < ∞
d [y] ← -1
data:image/s3,"s3://crabby-images/e9ee7/e9ee73b9ece1d7f23b138455ac48c78f0a26b437" alt="有向无环图中的单源最短路径"
adj [y] → r
-1 + 4 < 4
3 <4
d [r] ← 3
data:image/s3,"s3://crabby-images/c55b6/c55b6af683c34933c8280fd77997b1f9bd168b49" alt="有向无环图中的单源最短路径"
因此, 最短路径是:
s to x is 2
s to y is -1
s to t is 3
s to r is 3