-
Notifications
You must be signed in to change notification settings - Fork 43
/
Copy pathdijkstra.py
141 lines (101 loc) · 3.65 KB
/
dijkstra.py
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
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
'''This code is largely based on K Hong's "DIJKSTRA'S SHORTEST PATH ALGORITHM",
http://www.bogotobogo.com/python/python_Dijkstras_Shortest_Path_Algorithm.php'''
import heapq, math
class Vertex:
def __init__(self, node):
self.id = node
self.adjacent = {}
self.distance = math.inf
self.visited = False
self.previous = None
def __lt__(self, vertex2):
return self.id < vertex2.id
def __le__(self, vertex2):
return self.id <= vertex2.id
def __gt__(self, vertex2):
return self.id > vertex2.id
def __ge__(self, vertex2):
return self.id >= vertex2.id
def addNeighbor(self, neighbor, weight=0):
self.adjacent[neighbor] = weight
def getNeighbors(self):
return self.adjacent.keys()
def getId(self):
return self.id
def getWeight(self, neighbor):
return self.adjacent[neighbor]
def setDistance(self, dist):
self.distance = dist
def getDistance(self):
return self.distance
def setPrevious(self, prev):
self.previous = prev
def setVisited(self):
self.visited = True
def __str__(self):
return str(self.id) + ' adjacent: ' + str([x.id for x in self.adjacent])
class Graph:
def __init__(self):
self.vertices = {}
self.nVertices = 0
def __iter__(self):
return iter(self.vertices.values())
def addVertex(self, node):
self.nVertices += 1
newVertex = Vertex(node)
self.vertices[node] = newVertex
return newVertex
def getVertex(self, n):
if n in self.vertices:
return self.vertices[n]
return None
def addEdge(self, source, dest, cost=0):
if source not in self.vertices:
self.addVertex(source)
if dest not in self.vertices:
self.addVertex(dest)
self.vertices[source].addNeighbor(self.vertices[dest], cost)
self.vertices[dest].addNeighbor(self.vertices[source], cost)
def getVertices(self):
return self.vertices.keys()
def setPrevious(self, current):
self.previous = current
def getPrevious(self, current):
return self.previous
def dijkstra(aGraph, start, target):
# Set the distance for the start node to zero
start.setDistance(0)
# Put tuple pair into the priority queue
unvisited = [(v.getDistance(),v) for v in aGraph]
heapq.heapify(unvisited)
while len(unvisited):
# Pops a vertex with the smallest distance
uv = heapq.heappop(unvisited)
current = uv[1]
current.setVisited()
#for next in v.adjacent:
for next in current.adjacent:
# if visited, skip
if next.visited:
continue
newDist = current.getDistance() + current.getWeight(next)
if newDist < next.getDistance():
next.setDistance(newDist)
next.setPrevious(current)
print('updated : current = %s next = %s new_dist = %s' \
%(current.getId(), next.getId(), next.getDistance()))
else:
print('not updated : current = %s next = %s new_dist = %s' \
%(current.getId(), next.getId(), next.getDistance()))
# Rebuild heap
# 1. Pop every item
while len(unvisited):
heapq.heappop(unvisited)
# 2. Put all vertices not visited into the queue
unvisited = [(v.getDistance(),v) for v in aGraph if not v.visited]
heapq.heapify(unvisited)
def shortest(v, path):
if v.previous:
path.append(v.previous.getId())
shortest(v.previous, path)
return