Skip to content

Commit

Permalink
Update DijkstrasAlgorithm.rst
Browse files Browse the repository at this point in the history
  • Loading branch information
Natalie committed Oct 22, 2014
1 parent f92d2ec commit f6013c4
Showing 1 changed file with 5 additions and 5 deletions.
10 changes: 5 additions & 5 deletions pythonds/source/Graphs/DijkstrasAlgorithm.rst
Original file line number Diff line number Diff line change
Expand Up @@ -10,9 +10,9 @@
Алгоритм Дейкстры
~~~~~~~~~~~~~~~~~

Для поиска кратчайшего пути мы собираемся использовать так называемый "алгоритм Дейкстры". Он является итеративным и возвращает кратчайшее расстояние от конкретного стартового узла до всех прочих узлов графа - результат, снова похожий на поиск в ширину.
Для поиска кратчайшего пути мы собираемся использовать так называемый "алгоритм Дейкстры". Он является итеративным и возвращает кратчайшее расстояние от конкретного стартового узла до всех прочих узлов графа - очень похоже на результат поиска в ширину.

Чтобы отслеживать общие затраты на продвижение от начального узла до каждого конечного пункта, мы используем поле ``dist`` класса ``Vertex``. Оно будет содержать текущий общий вес кратчайшего пути от старта до запрашиваемой вершины. Алгоритм повторяется для каждого узла в графе, однако, последовательность итераций задаётся очередью с приоритетом. Значение, используемое для определения порядка объектов в ней, - ``dist``. Когда вершина только создаётся, ``dist`` устанавливается в очень большую величину. Теоретически ею должна быть бесконечность, но на практике мы просто выбираем число, большее любого реального расстояния, которое будет использоваться при решении задачи.
Чтобы отслеживать общие затраты на продвижение от начального узла до каждого конечного пункта, мы используем поле ``dist`` класса ``Vertex``. Оно будет содержать текущий общий вес кратчайшего пути от старта до запрашиваемой вершины. Алгоритм повторяется для каждого узла в графе, однако, последовательность итераций задаётся очередью с приоритетом. Для определения порядка объектов в ней используется значение ``dist``. Когда вершина только создаётся, ``dist`` устанавливается в очень большую величину. Теоретически ею должна быть бесконечность, но на практике мы просто выбираем число, большее любого реального расстояния, которое будет использоваться при решении задачи.

Код алгоритма Дейкстры показан в :ref:`листинге 1 <lst_shortpath>`. Результатом его работы станут правильно установленные расстояния, а также ссылки на предшественника для каждой вершины графа.

Expand All @@ -37,13 +37,13 @@
nextVert.setPred(currentVert)
pq.decreaseKey(nextVert,newDist)

Алгоритм Дейкстры использует очередь с приоритетом. Вы можете помнить, что она основывается на куче, которую мы реализовали в главе, посвящённой деревьям. Однако, между тем простым вариантом и реализацией для алгоритма Дейкстры есть несколько отличий. Во-первых, класс ``PriorityQueue`` сохраняет кортежи пар ключ-значение. Это очень важно для алгоритма Дейкстры, поскольку ключи в очереди с приоритетом должны быть связаны с ключами вершин графа. Во-вторых, значение используется для определения приоритета, а следовательно - для определения позиции ключа в очереди. В этой реализации мы используем растояние до вершины как приоритет, покольку (как увидим дальше) для исследования мы всегда будем стремиться найти узел с наименьшим расстоянием. Следующее отличие - в дополнительном методе ``decreaseKey``. Как вы можете видеть, он используется, когда нужно уменьшить расстояние до вершины, уже находящейся в очереди, т.е. переместить её ближе к началу.
Алгоритм Дейкстры использует очередь с приоритетом. Вы можете помнить, что она основывается на куче, которую мы создали в главе, посвящённой деревьям. Однако, между тем простым вариантом и реализацией для алгоритма Дейкстры есть несколько отличий. Во-первых, класс ``PriorityQueue`` сохраняет кортежи пар ключ-значение. Это очень важно для алгоритма Дейкстры, поскольку ключи в очереди с приоритетом должны быть связаны с ключами вершин графа. Во-вторых, значение используется для расстановки приоритетов, а следовательно и для определения позиции ключа в очереди. В этой реализации в качестве индикатора приоритета используется расстояние до вершины, покольку (как будет показано далее) для исследования мы всегда будем стремиться найти узел с наименьшим расстоянием. Следующее отличие между предыдущей и новой версиями - в дополнительном методе ``decreaseKey``. Как вы можете видеть, он используется, когда нужно уменьшить расстояние до вершины, уже стоящей в очереди, т.е. переместить её ближе к началу.

Давайте пройдём по алгоритму Дейкстры для одного из узлов, используя в качестве путеводителя нижеследующие рисунки. Начнём с вершины :math:`u`. С нею смежны три узла: :math:`v, w` и :math:`x`. Поскольку начальные расстояния для них инициализированы ``sys.maxint``, то расстояниями пути к ним от стартового узла станут их непосредственные веса. Вместе с обновлением издержек мы устанавливаем предшественником каждого узел :math:`u` и добавляем их в очередь. Расстояние будет использоваться как ключ приоритета. Состояние алгоритма на этот момент показано на :ref:`рисунке 3 <fig_dija>`.

На следующей итерации цикла ``while`` мы проверяем вершины, смежные с :math:`x`. Она выбрана следующей, поскольку имеет наименьшее значение ``dist`` и, следовательно, оказалась наверху очереди с приоритетом. Для :math:`x` мы рассматриваем её соседей :math:`u, v, w` и :math:`y`. Для каждого из них расчитываем путь через :math:`x` и смотрим, у кого он будет меньше существующей величины ``dist``. Очевидно, что это вариант :math:`y`, поскольку его расстояние - ``sys.maxint``, а не :math:`u` или :math:`v`, поскольку их значения 0 и 2 соответственно. Как бы то ни было, теперь мы знаем, что расстояние до :math:`w` будет наименьшим, если идти к ней через :math:`x`, а не непосредственно от :math:`u` к :math:`w`. Таким образом, мы обновляем расстояние :math:`w` и изменяем её предшественника с :math:`u` на :math:`x`. См. :ref:`рисунок 4 <fig_dijb>`, где показано состояние всех вершин на данный момент.

Следующим шагом станет рассмотрение соседей :math:`v` (см. :ref:`рисунок 5 <fig_dijc>`). Результаты этого шага не изменят граф, так что мы переходим к узлу :math:`y`. Для него (см. :ref:`рисунок 6 <fig_dijd>`) мы находим, что дешевле пройти через :math:`w` и :math:`z`, в связи с чем регулируем соответствующие расстояния и ссылки на предшественников. Наконец, проверяем узлы :math:`w` и :math:`z` (см. :ref:`рисунок 7 <fig_dije>` и :ref:`рисунок 8 <fig_dijf>`). Однако, никаких дополнительных изменений это не вносит, очередь с приоритетом пуста и алгоритм Дейкстры заканчивает свою работу.
Следующим шагом станет рассмотрение соседей :math:`v` (см. :ref:`рисунок 5 <fig_dijc>`). Его результаты не изменят граф, так что мы переходим к узлу :math:`y`. Для него (см. :ref:`рисунок 6 <fig_dijd>`) мы находим, что дешевле пройти через :math:`w` и :math:`z`, в связи с чем регулируем соответствующие расстояния и ссылки на предшественников. Наконец, проверяем узлы :math:`w` и :math:`z` (см. :ref:`рисунок 7 <fig_dije>` и :ref:`рисунок 8 <fig_dijf>`). Однако, никаких дополнительных изменений это не вносит, очередь с приоритетом пуста и алгоритм Дейкстры заканчивает свою работу.

.. _fig_dija:

Expand Down Expand Up @@ -89,4 +89,4 @@

Важно отметить, что алгоритм Дейкстры работает только в том случае, когда веса рёбер положительны. Вы можете самостоятельно проверить, что если хотя бы у одного из них будет отрицательный вес, то алгоритм никогда не завершится.

Отметим, что для прокладывания маршрута сообщения через интернет используются и другие алгоритмы поиска кратчайшего пути. Одной из проблем с алгоритмом Дейкстры для Всемирной паутины является необходимость иметь полное представление графа. Только в этом случае он может быть запущен. Следствием этого будет то, что каждый маршрутизатор должен обладать полной картой всех маршрутизаторов в интернете. На практике такое не встречается, поэтому существуют дргие варианты алгоритма, позволяющие маршрутизатору открывать граф по ходу дела. Одни из таких алгоритмов, о котором вы, возможно, захотите почитать, называется "дистанционно-векторный алгоритм маршрутизации".
Отметим, что для прокладывания маршрута сообщения через интернет используются и другие алгоритмы поиска кратчайшего пути. Одной из проблем с алгоритмом Дейкстры для Всемирной паутины является необходимость иметь полное представление графа. Только в этом случае он может быть запущен. Следствием этого будет то, что каждый маршрутизатор должен обладать полной картой всех маршрутизаторов в интернете. На практике такое не встречается, поэтому существуют дргие варианты алгоритма, позволяющие маршрутизатору открывать граф по ходу дела. Один из них, о котором вы, возможно, захотите почитать, называется "дистанционно-векторный алгоритм маршрутизации".

0 comments on commit f6013c4

Please sign in to comment.