2019-02-18 23:35:46 +08:00
|
|
|
"""
|
2020-05-22 14:10:11 +08:00
|
|
|
pseudo-code
|
|
|
|
|
2019-07-17 07:09:53 +08:00
|
|
|
DIJKSTRA(graph G, start vertex s, destination vertex d):
|
|
|
|
|
|
|
|
//all nodes initially unexplored
|
|
|
|
|
|
|
|
1 - let H = min heap data structure, initialized with 0 and s [here 0 indicates
|
|
|
|
the distance from start vertex s]
|
|
|
|
2 - while H is non-empty:
|
|
|
|
3 - remove the first node and cost of H, call it U and cost
|
|
|
|
4 - if U has been previously explored:
|
|
|
|
5 - go to the while loop, line 2 //Once a node is explored there is no need
|
|
|
|
to make it again
|
|
|
|
6 - mark U as explored
|
|
|
|
7 - if U is d:
|
|
|
|
8 - return cost // total cost from start to destination vertex
|
|
|
|
9 - for each edge(U, V): c=cost of edge(U,V) // for V in graph[U]
|
|
|
|
10 - if V explored:
|
|
|
|
11 - go to next V in line 9
|
|
|
|
12 - total_cost = cost + c
|
|
|
|
13 - add (total_cost,V) to H
|
|
|
|
|
|
|
|
You can think at cost as a distance where Dijkstra finds the shortest distance
|
2020-01-18 20:24:33 +08:00
|
|
|
between vertices s and v in a graph G. The use of a min heap as H guarantees
|
2019-07-17 07:09:53 +08:00
|
|
|
that if a vertex has already been explored there will be no other path with
|
|
|
|
shortest distance, that happens because heapq.heappop will always return the
|
|
|
|
next vertex with the shortest distance, considering that the heap stores not
|
|
|
|
only the distance between previous vertex and current vertex but the entire
|
|
|
|
distance between each vertex that makes up the path from start vertex to target
|
|
|
|
vertex.
|
2019-02-18 23:35:46 +08:00
|
|
|
"""
|
|
|
|
import heapq
|
|
|
|
|
|
|
|
|
|
|
|
def dijkstra(graph, start, end):
|
2020-01-18 20:24:33 +08:00
|
|
|
"""Return the cost of the shortest path between vertices start and end.
|
2019-07-17 07:09:53 +08:00
|
|
|
|
|
|
|
>>> dijkstra(G, "E", "C")
|
|
|
|
6
|
|
|
|
>>> dijkstra(G2, "E", "F")
|
|
|
|
3
|
|
|
|
>>> dijkstra(G3, "E", "F")
|
|
|
|
3
|
|
|
|
"""
|
|
|
|
|
2019-02-18 23:35:46 +08:00
|
|
|
heap = [(0, start)] # cost from start node,end node
|
2019-06-23 23:32:12 +08:00
|
|
|
visited = set()
|
2019-02-18 23:35:46 +08:00
|
|
|
while heap:
|
|
|
|
(cost, u) = heapq.heappop(heap)
|
|
|
|
if u in visited:
|
|
|
|
continue
|
2019-06-23 23:32:12 +08:00
|
|
|
visited.add(u)
|
2019-02-18 23:35:46 +08:00
|
|
|
if u == end:
|
|
|
|
return cost
|
2019-07-17 07:09:53 +08:00
|
|
|
for v, c in graph[u]:
|
2019-02-18 23:35:46 +08:00
|
|
|
if v in visited:
|
|
|
|
continue
|
|
|
|
next = cost + c
|
|
|
|
heapq.heappush(heap, (next, v))
|
2019-07-17 07:09:53 +08:00
|
|
|
return -1
|
|
|
|
|
|
|
|
|
|
|
|
G = {
|
|
|
|
"A": [["B", 2], ["C", 5]],
|
|
|
|
"B": [["A", 2], ["D", 3], ["E", 1], ["F", 1]],
|
|
|
|
"C": [["A", 5], ["F", 3]],
|
|
|
|
"D": [["B", 3]],
|
|
|
|
"E": [["B", 4], ["F", 3]],
|
|
|
|
"F": [["C", 3], ["E", 3]],
|
|
|
|
}
|
|
|
|
|
2019-07-21 00:31:08 +08:00
|
|
|
r"""
|
2019-07-17 07:09:53 +08:00
|
|
|
Layout of G2:
|
|
|
|
|
|
|
|
E -- 1 --> B -- 1 --> C -- 1 --> D -- 1 --> F
|
|
|
|
\ /\
|
|
|
|
\ ||
|
|
|
|
----------------- 3 --------------------
|
|
|
|
"""
|
|
|
|
G2 = {
|
|
|
|
"B": [["C", 1]],
|
|
|
|
"C": [["D", 1]],
|
|
|
|
"D": [["F", 1]],
|
|
|
|
"E": [["B", 1], ["F", 3]],
|
|
|
|
"F": [],
|
|
|
|
}
|
|
|
|
|
2019-07-21 00:31:08 +08:00
|
|
|
r"""
|
2019-07-17 07:09:53 +08:00
|
|
|
Layout of G3:
|
|
|
|
|
|
|
|
E -- 1 --> B -- 1 --> C -- 1 --> D -- 1 --> F
|
|
|
|
\ /\
|
|
|
|
\ ||
|
|
|
|
-------- 2 ---------> G ------- 1 ------
|
|
|
|
"""
|
|
|
|
G3 = {
|
|
|
|
"B": [["C", 1]],
|
|
|
|
"C": [["D", 1]],
|
|
|
|
"D": [["F", 1]],
|
|
|
|
"E": [["B", 1], ["G", 2]],
|
|
|
|
"F": [],
|
|
|
|
"G": [["F", 1]],
|
|
|
|
}
|
|
|
|
|
2022-10-13 06:54:20 +08:00
|
|
|
short_distance = dijkstra(G, "E", "C")
|
|
|
|
print(short_distance) # E -- 3 --> F -- 3 --> C == 6
|
2019-02-18 23:35:46 +08:00
|
|
|
|
2022-10-13 06:54:20 +08:00
|
|
|
short_distance = dijkstra(G2, "E", "F")
|
|
|
|
print(short_distance) # E -- 3 --> F == 3
|
2019-02-18 23:35:46 +08:00
|
|
|
|
2022-10-13 06:54:20 +08:00
|
|
|
short_distance = dijkstra(G3, "E", "F")
|
|
|
|
print(short_distance) # E -- 2 --> G -- 1 --> F == 3
|
2019-02-18 23:35:46 +08:00
|
|
|
|
2019-07-17 07:09:53 +08:00
|
|
|
if __name__ == "__main__":
|
|
|
|
import doctest
|
2019-10-05 13:14:13 +08:00
|
|
|
|
2019-07-17 07:09:53 +08:00
|
|
|
doctest.testmod()
|