-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathch18_graphs.py
276 lines (212 loc) · 9.96 KB
/
ch18_graphs.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
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
"""
Chapter 18
Connecting Everything with Graphs
Exercises
The following exercises provide you with the opportunity to practice with graphs.
"""
from queue import Queue
class Vertex:
def __init__(self, value):
self.value = value
self.adjacent_vertices = []
def add_adjacent_vertex(self, vertex):
self.adjacent_vertices.append(vertex)
def question4():
"""
In this chapter, I only provided the code for breadth-first traversal, as discussed in Breadth-First Search.
That is, the code simply printed the value of each vertex.
Modify the code so that it performs an actual search for a vertex value provided to the function.
(We did this for depth-first search.)
That is, if the function finds the vertex it's searching for, it should return that vertex's value.
Otherwise, it should return null
"""
def bfs_lookup(source_vertex, vertex_to_find):
queue = Queue()
visited_vertices = {}
visited_vertices[source_vertex.value] = True
queue.put(source_vertex)
# While the queue is not empty:
while queue.qsize():
# Remove the first vertex off the queue and make it the current vertex:
current_vertex = queue.get()
# Print the current vertex's value:
print(current_vertex.value, end=" ")
if current_vertex.value == vertex_to_find.value:
return current_vertex.value
# Iterate over current vertex's adjacent vertices:
for adjacent_vertex in current_vertex.adjacent_vertices:
# If we have not yet visited the adjacent vertex:
if not visited_vertices.get(adjacent_vertex.value):
# Mark the adjacent vertex as visited:
visited_vertices[adjacent_vertex.value] = True
# Add the adjacent vertex to the queue:
queue.put(adjacent_vertex)
# Vertex not found
return None
###############
A = Vertex("A")
B = Vertex("B")
C = Vertex("C")
D = Vertex("D")
E = Vertex("E")
F = Vertex("F")
G = Vertex("G")
H = Vertex("H")
I = Vertex("I")
J = Vertex("J")
K = Vertex("K")
L = Vertex("L")
M = Vertex("M")
N = Vertex("N")
O = Vertex("O")
P = Vertex("P")
Z = Vertex("Z")
A.add_adjacent_vertex(B)
A.add_adjacent_vertex(C)
A.add_adjacent_vertex(D)
B.add_adjacent_vertex(E)
B.add_adjacent_vertex(F)
E.add_adjacent_vertex(J)
F.add_adjacent_vertex(J)
J.add_adjacent_vertex(O)
C.add_adjacent_vertex(G)
G.add_adjacent_vertex(K)
D.add_adjacent_vertex(H)
D.add_adjacent_vertex(I)
H.add_adjacent_vertex(L)
H.add_adjacent_vertex(M)
I.add_adjacent_vertex(M)
I.add_adjacent_vertex(N)
N.add_adjacent_vertex(P)
vertex_to_find = O
if found := bfs_lookup(A, vertex_to_find):
print(f"\nBFS found vertex: {found}")
else:
print(f"\nBFS did not found vertex: {vertex_to_find.value}")
def question5():
"""
In Dijkstra's Algorithm, we saw how Dijkstra's algorithm helped us find the shortest path within a weighted graph.
However, the concept of a shortest path exists within an unweighted graph as well. How?
The shortest path in a classic (unweighted) graph is the path that takes the fewest number of vertices to get from one vertex to another.
This can be especially useful in social networking applications. Take the example network that follows:
If we want to know how Idris is to connected Lina, we'd see that she's connected to her from two different directions.
That is, Idris is a second-degree connection to Lina through Kamil, but she is also a fifth-degree connection through Talia.
Now, we're probably interested in how closely Idris is connected to Lina,
so the fact that she's a fifth-degree connection is unimportant given that they're also second-degree connections.
Write a function that accepts two vertices from a graph and returns the shortest path between them.
The function should return an array containing the precise path, such as ["Idris", "Kamil", "Lina"].
Hint: The algorithm may contain elements of both breadth-first search and Dijkstra's algorithm.
"""
def dijkstra_shortest_path(source_vertex, destination_vertex):
shortest_path_table = {}
shortest_previous_middle_path_table = {}
# To keep our code simple, we'll use a basic array to keep track of
# the known cities we haven't yet visited:
unvisited_vertex = []
# We keep track of the cities we've visited using a hash table.
# We could have used an array, but since we'll be doing lookups,
# a hash table is more efficient:
visited_vertex = {}
# We add the starting city's name as the first key inside the
# shortest_path_table. It has a value of 0, since it costs nothing
# to get there:
shortest_path_table[source_vertex.value] = 0
current_vertex = source_vertex
# This loop is the core of the algorithm. It runs as long as we can
# visit a city that we haven't visited yet:
while current_vertex:
# We add the current_vertex's name to the visited_vertex hash to record
# that we've officially visited it. We also remove it from the list
# of unvisited cities:
visited_vertex[current_vertex.value] = True
if current_vertex in unvisited_vertex:
unvisited_vertex.remove(current_vertex)
# We iterate over each of the current_vertex's adjacent cities:
for adjacent_vertexs in current_vertex.adjacent_vertices:
# If we've discovered a new city,
# we add it to the list of unvisited_vertex:
if not visited_vertex.get(adjacent_vertexs.value):
unvisited_vertex.append(adjacent_vertexs)
# We calculate the price of getting from the STARTING city to the
# ADJACENT city using the CURRENT city as the second-to-last stop:
price_through_current_city = shortest_path_table[current_vertex.value] + 1
# If the price from the STARTING city to the ADJACENT city is
# the cheapest one we've found so far...
if not shortest_path_table.get(adjacent_vertexs.value) or price_through_current_city < shortest_path_table[adjacent_vertexs.value]:
# ... we update our two tables:
shortest_path_table[adjacent_vertexs.value] = price_through_current_city
shortest_previous_middle_path_table[adjacent_vertexs.value] = current_vertex.value
# We visit our next unvisited city. We choose the one that is cheapest
# to get to from the STARTING city:
if unvisited_vertex:
current_vertex = unvisited_vertex[0]
for vertex in unvisited_vertex:
if shortest_path_table[vertex.value] < shortest_path_table[current_vertex.value]:
current_vertex = vertex
else:
current_vertex = None
# We have completed the core algorithm. At this point, the
# shortest_path_table contains all the cheapest prices to get to each
# city from the starting city. However, to calculate the precise path
# to take from our starting city to our final destination, we need to move on.
# We'll build the shortest path using a simple array:
shortest_path = []
# To construct the shortest path, we need to work backwards from our
# final destination. So we begin with the final destination as our
# current_vertex_value:
current_vertex_value = destination_vertex.value
# We loop until we reach our starting city:
while current_vertex_value != source_vertex.value:
# We add each current_vertex_value we encounter to the shortest path array:
shortest_path.append(current_vertex_value)
# We use the shortest_previous_middle_path_table to follow each city
# to its previous stopover city:
current_vertex_value = shortest_previous_middle_path_table[current_vertex_value]
# We cap things off by adding the starting city to the shortest path:
shortest_path.append(source_vertex.value)
# We reverse the output so we can see the path from beginning to end:
shortest_path.reverse()
return shortest_path
A = Vertex("A")
B = Vertex("B")
C = Vertex("C")
D = Vertex("D")
E = Vertex("E")
F = Vertex("F")
G = Vertex("G")
H = Vertex("H")
I = Vertex("I")
J = Vertex("J")
K = Vertex("K")
L = Vertex("L")
M = Vertex("M")
N = Vertex("N")
O = Vertex("O")
P = Vertex("P")
Z = Vertex("Z")
A.add_adjacent_vertex(B)
A.add_adjacent_vertex(C)
A.add_adjacent_vertex(D)
B.add_adjacent_vertex(E)
B.add_adjacent_vertex(F)
E.add_adjacent_vertex(J)
F.add_adjacent_vertex(J)
J.add_adjacent_vertex(O) # ['A', 'D', 'I', 'N', 'P', 'O'] if commented. A to O
C.add_adjacent_vertex(G)
G.add_adjacent_vertex(K)
D.add_adjacent_vertex(H)
D.add_adjacent_vertex(I)
H.add_adjacent_vertex(L)
H.add_adjacent_vertex(M)
I.add_adjacent_vertex(M)
I.add_adjacent_vertex(N)
#N.add_adjacent_vertex(P)
P.add_adjacent_vertex(O)
O.add_adjacent_vertex(P)
source_vertex = A
destination_vertex = P
#if shortest_path := find_shortest_path(source_vertex, destination_vertex):
if shortest_path := dijkstra_shortest_path(source_vertex, destination_vertex):
print(f"\nThe Shortest path between {source_vertex.value} and {destination_vertex.value} is {shortest_path}")
#question4()
question5()