Это казалось забавной проблемой, поэтому я решил попробовать.Вот моя попытка.Вы не сказали, в каком формате находится ваш график, поэтому я угадал словарь, подобный следующему:
example_graph = {0:[1,2], 1:[0,3,4], 3:[1], 4:[1], 2:[0,5,6], 5:[2], 6:[2]}
Код просматривает список активных в данный момент вершин, начиная с указанных начальных и целевых вершин,и помнит, как он дошел до них через словарь {vertices: lists}.Если активная вершина оказывается рядом с другой активной вершиной с путем, который начинается с другого конца, она объединяет списки и возвращает результаты, в противном случае она расширяет все пути и продолжается.
def bi_directional_search(graph, start, goal):
# Check if start and goal are equal.
if start == goal:
return [start]
# Get dictionary of currently active vertices with their corresponding paths.
active_vertices_path_dict = {start: [start], goal: [goal]}
# Vertices we have already examined.
inactive_vertices = set()
while len(active_vertices_path_dict) > 0:
# Make a copy of active vertices so we can modify the original dictionary as we go.
active_vertices = list(active_vertices_path_dict.keys())
for vertex in active_vertices:
# Get the path to where we are.
current_path = active_vertices_path_dict[vertex]
# Record whether we started at start or goal.
origin = current_path[0]
# Check for new neighbours.
current_neighbours = set(graph[vertex]) - inactive_vertices
# Check if our neighbours hit an active vertex
if len(current_neighbours.intersection(active_vertices)) > 0:
for meeting_vertex in current_neighbours.intersection(active_vertices):
# Check the two paths didn't start at same place. If not, then we've got a path from start to goal.
if origin != active_vertices_path_dict[meeting_vertex][0]:
# Reverse one of the paths.
active_vertices_path_dict[meeting_vertex].reverse()
# return the combined results
return active_vertices_path_dict[vertex] + active_vertices_path_dict[meeting_vertex]
# No hits, so check for new neighbours to extend our paths.
if len(set(current_neighbours) - inactive_vertices - set(active_vertices)) == 0:
# If none, then remove the current path and record the endpoint as inactive.
active_vertices_path_dict.pop(vertex, None)
inactive_vertices.add(vertex)
else:
# Otherwise extend the paths, remove the previous one and update the inactive vertices.
for neighbour_vertex in current_neighbours - inactive_vertices - set(active_vertices):
active_vertices_path_dict[neighbour_vertex] = current_path + [neighbour_vertex]
active_vertices.append(neighbour_vertex)
active_vertices_path_dict.pop(vertex, None)
inactive_vertices.add(vertex)
return None