Двунаправленный поиск - PullRequest
       23

Двунаправленный поиск

0 голосов
/ 30 января 2019

Я пытаюсь реализовать двунаправленный поиск в Python.

Как я понимаю, я должен каким-то образом объединить два поиска в ширину, один из которых начинается с начального (или корневого) узла, а другой - с целевого (или конечного) узла.Двунаправленный поиск заканчивается, когда оба поиска в ширину «встречаются» в одной и той же вершине.Я реализовал BFS, код приведен ниже

def bfs(graph, start):
    path = []
    queue = [start]
    while queue:
        vertex = queue.pop(0)
        if vertex not in path:
            path.append(vertex)
            queue.extend(graph[vertex])
    return path

Не могли бы вы предоставить мне пример кода (на Python) или ссылку с кодом для поиска двунаправленного графика?

1 Ответ

0 голосов
/ 30 января 2019

Это казалось забавной проблемой, поэтому я решил попробовать.Вот моя попытка.Вы не сказали, в каком формате находится ваш график, поэтому я угадал словарь, подобный следующему:

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
...