2019-12-19 17:40:16 +08:00
|
|
|
"""Breadth-first search shortest path implementations.
|
|
|
|
doctest:
|
|
|
|
python -m doctest -v bfs_shortest_path.py
|
|
|
|
Manual test:
|
|
|
|
python bfs_shortest_path.py
|
|
|
|
"""
|
2021-06-29 19:44:35 +08:00
|
|
|
demo_graph = {
|
2019-10-05 13:14:13 +08:00
|
|
|
"A": ["B", "C", "E"],
|
|
|
|
"B": ["A", "D", "E"],
|
|
|
|
"C": ["A", "F", "G"],
|
|
|
|
"D": ["B"],
|
|
|
|
"E": ["A", "B", "D"],
|
|
|
|
"F": ["C"],
|
|
|
|
"G": ["C"],
|
|
|
|
}
|
|
|
|
|
2019-05-17 11:12:06 +08:00
|
|
|
|
2021-06-29 19:44:35 +08:00
|
|
|
def bfs_shortest_path(graph: dict, start, goal) -> list[str]:
|
2019-12-19 17:40:16 +08:00
|
|
|
"""Find shortest path between `start` and `goal` nodes.
|
2020-09-10 16:31:26 +08:00
|
|
|
Args:
|
|
|
|
graph (dict): node/list of neighboring nodes key/value pairs.
|
|
|
|
start: start node.
|
|
|
|
goal: target node.
|
|
|
|
Returns:
|
|
|
|
Shortest path between `start` and `goal` nodes as a string of nodes.
|
|
|
|
'Not found' string if no path found.
|
|
|
|
Example:
|
2021-06-29 19:44:35 +08:00
|
|
|
>>> bfs_shortest_path(demo_graph, "G", "D")
|
2020-09-10 16:31:26 +08:00
|
|
|
['G', 'C', 'A', 'B', 'D']
|
2021-06-29 19:44:35 +08:00
|
|
|
>>> bfs_shortest_path(demo_graph, "G", "G")
|
|
|
|
['G']
|
|
|
|
>>> bfs_shortest_path(demo_graph, "G", "Unknown")
|
|
|
|
[]
|
2019-12-19 17:40:16 +08:00
|
|
|
"""
|
2019-05-17 11:12:06 +08:00
|
|
|
# keep track of explored nodes
|
2020-11-21 14:58:52 +08:00
|
|
|
explored = set()
|
2019-05-17 11:12:06 +08:00
|
|
|
# keep track of all the paths to be checked
|
|
|
|
queue = [[start]]
|
2019-10-05 13:14:13 +08:00
|
|
|
|
2019-05-17 11:12:06 +08:00
|
|
|
# return path if start is goal
|
|
|
|
if start == goal:
|
2021-06-29 19:44:35 +08:00
|
|
|
return [start]
|
2019-10-05 13:14:13 +08:00
|
|
|
|
2019-05-17 11:12:06 +08:00
|
|
|
# keeps looping until all possible paths have been checked
|
|
|
|
while queue:
|
|
|
|
# pop the first path from the queue
|
|
|
|
path = queue.pop(0)
|
|
|
|
# get the last node from the path
|
|
|
|
node = path[-1]
|
|
|
|
if node not in explored:
|
|
|
|
neighbours = graph[node]
|
|
|
|
# go through all neighbour nodes, construct a new path and
|
|
|
|
# push it into the queue
|
|
|
|
for neighbour in neighbours:
|
|
|
|
new_path = list(path)
|
|
|
|
new_path.append(neighbour)
|
|
|
|
queue.append(new_path)
|
|
|
|
# return path if neighbour is goal
|
|
|
|
if neighbour == goal:
|
|
|
|
return new_path
|
2019-10-05 13:14:13 +08:00
|
|
|
|
2019-05-17 11:12:06 +08:00
|
|
|
# mark node as explored
|
2020-11-21 14:58:52 +08:00
|
|
|
explored.add(node)
|
2019-10-05 13:14:13 +08:00
|
|
|
|
2019-05-17 11:12:06 +08:00
|
|
|
# in case there's no path between the 2 nodes
|
2021-06-29 19:44:35 +08:00
|
|
|
return []
|
2019-10-05 13:14:13 +08:00
|
|
|
|
|
|
|
|
2019-12-19 17:40:16 +08:00
|
|
|
def bfs_shortest_path_distance(graph: dict, start, target) -> int:
|
|
|
|
"""Find shortest path distance between `start` and `target` nodes.
|
2020-09-10 16:31:26 +08:00
|
|
|
Args:
|
|
|
|
graph: node/list of neighboring nodes key/value pairs.
|
|
|
|
start: node to start search from.
|
|
|
|
target: node to search for.
|
|
|
|
Returns:
|
|
|
|
Number of edges in shortest path between `start` and `target` nodes.
|
|
|
|
-1 if no path exists.
|
|
|
|
Example:
|
2021-06-29 19:44:35 +08:00
|
|
|
>>> bfs_shortest_path_distance(demo_graph, "G", "D")
|
2020-09-10 16:31:26 +08:00
|
|
|
4
|
2021-06-29 19:44:35 +08:00
|
|
|
>>> bfs_shortest_path_distance(demo_graph, "A", "A")
|
2020-09-10 16:31:26 +08:00
|
|
|
0
|
2021-06-29 19:44:35 +08:00
|
|
|
>>> bfs_shortest_path_distance(demo_graph, "A", "Unknown")
|
2020-09-10 16:31:26 +08:00
|
|
|
-1
|
2019-12-19 17:40:16 +08:00
|
|
|
"""
|
|
|
|
if not graph or start not in graph or target not in graph:
|
|
|
|
return -1
|
|
|
|
if start == target:
|
|
|
|
return 0
|
|
|
|
queue = [start]
|
2020-11-21 14:58:52 +08:00
|
|
|
visited = set(start)
|
2019-12-19 17:40:16 +08:00
|
|
|
# Keep tab on distances from `start` node.
|
|
|
|
dist = {start: 0, target: -1}
|
|
|
|
while queue:
|
|
|
|
node = queue.pop(0)
|
|
|
|
if node == target:
|
|
|
|
dist[target] = (
|
|
|
|
dist[node] if dist[target] == -1 else min(dist[target], dist[node])
|
|
|
|
)
|
|
|
|
for adjacent in graph[node]:
|
|
|
|
if adjacent not in visited:
|
2020-11-21 14:58:52 +08:00
|
|
|
visited.add(adjacent)
|
2019-12-19 17:40:16 +08:00
|
|
|
queue.append(adjacent)
|
|
|
|
dist[adjacent] = dist[node] + 1
|
|
|
|
return dist[target]
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
2021-06-29 19:44:35 +08:00
|
|
|
print(bfs_shortest_path(demo_graph, "G", "D")) # returns ['G', 'C', 'A', 'B', 'D']
|
|
|
|
print(bfs_shortest_path_distance(demo_graph, "G", "D")) # returns 4
|