Алгоритм Флойда-Уоршалла

Алгоритм в теории графов
Алгоритм Флойда-Уоршалла
СортЗадача нахождения кратчайшего пути для всех пар вершин (для взвешенных графов)
Структура данныхГрафик
Худший вариант производительности Θ ( | В | 3 ) {\displaystyle \Theta (|V|^{3})}
Лучшая производительность Θ ( | В | 3 ) {\displaystyle \Theta (|V|^{3})}
Средняя производительность Θ ( | В | 3 ) {\displaystyle \Theta (|V|^{3})}
Наихудшая сложность пространства Θ ( | В | 2 ) {\displaystyle \Theta (|V|^{2})}

В информатике алгоритм Флойда–Уоршелла (также известный как алгоритм Флойда , алгоритм Роя–Уоршелла , алгоритм Роя–Флойда или алгоритм WFI ) — это алгоритм поиска кратчайших путей в ориентированном взвешенном графе с положительными или отрицательными весами ребер (но без отрицательных циклов). [1] [2] Однократное выполнение алгоритма найдет длины (суммарные веса) кратчайших путей между всеми парами вершин. Хотя он не возвращает детали самих путей, можно реконструировать пути с помощью простых модификаций алгоритма. Версии алгоритма также могут быть использованы для поиска транзитивного замыкания отношения или (в связи с системой голосования Шульце ) широчайших путей между всеми парами вершин во взвешенном графе. Р {\displaystyle R}

История и наименование

Алгоритм Флойда–Уоршалла является примером динамического программирования и был опубликован в его нынешней признанной форме Робертом Флойдом в 1962 году. [3] Однако, по сути, он такой же, как алгоритмы, ранее опубликованные Бернардом Роем в 1959 году [4] , а также Стивеном Уоршаллом в 1962 году [5] для нахождения транзитивного замыкания графа, [6] и тесно связан с алгоритмом Клини (опубликованным в 1956 году) для преобразования детерминированного конечного автомата в регулярное выражение . [7] Современная формулировка алгоритма в виде трех вложенных циклов for была впервые описана Питером Ингерманом также в 1962 году. [8]

Алгоритм

Алгоритм Флойда–Уоршелла сравнивает множество возможных путей по графу между каждой парой вершин. Он гарантированно находит все кратчайшие пути и может делать это с помощью сравнений в графе, [1] [9] даже если в графе могут быть ребра. Он делает это путем постепенного улучшения оценки кратчайшего пути между двумя вершинами, пока оценка не станет оптимальной. Θ ( | В | 3 ) {\displaystyle \Theta (|V|^{3})} Θ ( | В | 2 ) {\displaystyle \Theta (|V|^{2})}

Рассмотрим граф с вершинами, пронумерованными от 1 до  . Далее рассмотрим функцию , которая возвращает длину кратчайшего возможного пути (если он существует) от до , используя вершины только из множества в качестве промежуточных точек на пути. Теперь, учитывая эту функцию, наша цель — найти длину кратчайшего пути от каждого до каждого , используя любую вершину из . По определению, это значение , которое мы найдем рекурсивно . Г {\displaystyle G} В {\displaystyle V} Н {\displaystyle N} с час о г т е с т П а т час ( я , дж , к ) {\displaystyle \mathrm {shortestPath} (i,j,k)} я {\displaystyle я} дж {\displaystyle j} { 1 , 2 , , к } {\displaystyle \{1,2,\ldots ,k\}} я {\displaystyle я} дж {\displaystyle j} { 1 , 2 , , Н } {\displaystyle \{1,2,\ldots ,N\}} с час о г т е с т П а т час ( я , дж , Н ) {\displaystyle \mathrm {shortestPath} (i,j,N)}

Обратите внимание, что должно быть меньше или равно : у нас больше гибкости, если нам разрешено использовать вершину . Если на самом деле меньше , то должен быть путь от до с использованием вершин , который короче любого такого пути, не использующего вершину . Поскольку нет отрицательных циклов, этот путь можно разложить следующим образом: с час о г т е с т П а т час ( я , дж , к ) {\displaystyle \mathrm {shortestPath} (i,j,k)} с час о г т е с т П а т час ( я , дж , к 1 ) {\displaystyle \mathrm {shortestPath} (i,j,k-1)} к {\displaystyle к} с час о г т е с т П а т час ( я , дж , к ) {\displaystyle \mathrm {shortestPath} (i,j,k)} с час о г т е с т П а т час ( я , дж , к 1 ) {\displaystyle \mathrm {shortestPath} (i,j,k-1)} я {\displaystyle я} дж {\displaystyle j} { 1 , 2 , , к } {\displaystyle \{1,2,\ldots ,k\}} к {\displaystyle к}

(1) путь от до , который использует вершины , за которым следует я {\displaystyle я} к {\displaystyle к} { 1 , 2 , , к 1 } {\displaystyle \{1,2,\ldots ,k-1\}}
(2) путь от до , использующий вершины . к {\displaystyle к} дж {\displaystyle j} { 1 , 2 , , к 1 } {\displaystyle \{1,2,\ldots ,k-1\}}

И, конечно, это должен быть кратчайший такой путь (или несколько таких), иначе мы могли бы еще больше уменьшить длину. Другими словами, мы пришли к рекурсивной формуле:

с час о г т е с т П а т час ( я , дж , к ) = {\displaystyle \mathrm {shortestPath} (i,j,k)=}
м я н ( с час о г т е с т П а т час ( я , дж , к 1 ) , {\displaystyle \mathrm {min} {\Big (}\mathrm {shortestPath} (i,j,k-1),}
с час о г т е с т П а т час ( я , к , к 1 ) + с час о г т е с т П а т час ( к , дж , к 1 ) ) {\displaystyle \mathrm {shortestPath} (i,k,k-1)+\mathrm {shortestPath} (k,j,k-1){\Big)}} .

Базовый вариант определяется как

с час о г т е с т П а т час ( я , дж , 0 ) = ж ( я , дж ) , {\displaystyle \mathrm {shortestPath} (i,j,0)=w(i,j),}

где обозначает вес ребра от до , если оно существует, и ∞ (бесконечность) в противном случае. ж ( я , дж ) {\displaystyle w(i,j)} я {\displaystyle я} дж {\displaystyle j}

Эти формулы являются сердцем алгоритма Флойда–Уоршелла. Алгоритм работает, сначала вычисляя для всех пар для , затем , затем и так далее. Этот процесс продолжается до , и мы находим кратчайший путь для всех пар с использованием любых промежуточных вершин. Ниже приведен псевдокод для этой базовой версии. с час о г т е с т П а т час ( я , дж , к ) {\displaystyle \mathrm {shortestPath} (i,j,k)} ( я , дж ) {\displaystyle (я,j)} к = 0 {\displaystyle к=0} к = 1 {\displaystyle к=1} к = 2 {\displaystyle к=2} к = Н {\displaystyle к=Н} ( я , дж ) {\displaystyle (я,j)}

Псевдокод

пусть dist будет массивом минимальных расстояний размером |V| × |V|, инициализированным значением ∞ (бесконечность) для каждого ребра ( u , v ) do dist[ u ][ v ] ← w( u , v ) // Вес ребра ( u , v ) для каждой вершины v  do dist[ v ][ v ] ← 0 для  k  от 1 до |V| для  i  от 1 до |V| для  j  от 1 до |V| if dist[ i ][ j ] > dist[ i ][ k ] + dist[ k ][ j ] dist[ i ][ j ] ← dist[ i ][ k ] + dist[ k ][ j ] конец если

Пример

Приведенный выше алгоритм реализован на графике слева ниже:

До первой рекурсии внешнего цикла, обозначенного выше как k = 0 , единственные известные пути соответствуют одиночным ребрам в графе. При k = 1 находятся пути, проходящие через вершину 1: в частности, находится путь [2,1,3], заменяющий путь [2,3], который имеет меньше ребер, но длиннее (с точки зрения веса). При k = 2 находятся пути, проходящие через вершины {1,2}. Красные и синие поля показывают, как путь [4,2,1,3] собирается из двух известных путей [4,2] и [2,1,3], встреченных в предыдущих итерациях, с 2 в пересечении. Путь [4,2,3] не рассматривается, поскольку [2,1,3] является кратчайшим путем, встреченным до сих пор от 2 до 3. При k = 3 находятся пути, проходящие через вершины {1,2,3}. Наконец, при k = 4 находятся все кратчайшие пути.

Матрица расстояний на каждой итерации k , где обновленные расстояния выделены жирным шрифтом , будет иметь вид:

к = 0дж
1234
я10−2
2403
302
4−10
к = 1дж
1234
я10−2
2402
302
4−10
к = 2дж
1234
я10−2
2402
302
43−110
к = 3дж
1234
я10−20
24024
302
43−110
к = 4дж
1234
я10−1−20
24024
35102
43−110

Поведение с отрицательными циклами

Отрицательный цикл — это цикл, сумма ребер которого дает отрицательное значение. Не существует кратчайшего пути между любой парой вершин , которые образуют часть отрицательного цикла, поскольку длины путей от до могут быть сколь угодно малыми (отрицательными). Для численно значимых выходных данных алгоритм Флойда–Уоршелла предполагает, что отрицательных циклов нет. Тем не менее, если есть отрицательные циклы, алгоритм Флойда–Уоршелла может быть использован для их обнаружения. Интуиция такова: я {\displaystyle я} дж {\displaystyle j} я {\displaystyle я} дж {\displaystyle j}

  • Алгоритм Флойда–Уоршелла итеративно пересматривает длины путей между всеми парами вершин , включая , где ; ( я , дж ) {\displaystyle (я,j)} я = дж {\displaystyle i=j}
  • Первоначально длина пути равна нулю; ( я , я ) {\displaystyle (я,я)}
  • Путь может улучшить это значение только в том случае, если его длина меньше нуля, т.е. он обозначает отрицательный цикл; [ я , к , , я ] {\displaystyle [я,к,\ldots ,я]}
  • Таким образом, после алгоритма будет отрицательным, если существует путь отрицательной длины от обратно до . ( я , я ) {\displaystyle (я,я)} я {\displaystyle я} я {\displaystyle я}

Следовательно, для обнаружения отрицательных циклов с помощью алгоритма Флойда–Уоршелла можно проверить диагональ матрицы пути, и наличие отрицательного числа указывает на то, что граф содержит по крайней мере один отрицательный цикл. [9] Во время выполнения алгоритма, если есть отрицательный цикл, могут появляться экспоненциально большие числа, такие как , где — наибольшее абсолютное значение отрицательного ребра в графе. Чтобы избежать проблем переполнения/незаполнения, следует проверять наличие отрицательных чисел на диагонали матрицы пути во внутреннем цикле for алгоритма. [10] Очевидно, что в неориентированном графе отрицательное ребро создает отрицательный цикл (т. е. замкнутый обход), включающий его инцидентные вершины. Рассматривая все ребра приведенного выше примера графа как неориентированные, например, последовательность вершин 4 – 2 – 4 представляет собой цикл с суммой весов −2. Ω ( 6 н 1 ж м а х ) {\displaystyle \Омега (\cdot 6^{n-1}w_{макс})} ж м а х {\displaystyle w_{макс}}

Реконструкция пути

Алгоритм Флойда–Уоршелла обычно предоставляет только длины путей между всеми парами вершин. С помощью простых модификаций можно создать метод для реконструкции фактического пути между любыми двумя конечными вершинами. Хотя кто-то может быть склонен хранить фактический путь от каждой вершины до каждой другой вершины, это не обязательно и, по сути, очень затратно с точки зрения памяти. Вместо этого мы можем использовать дерево кратчайших путей , которое может быть вычислено для каждого узла со временем с использованием памяти и позволяет нам эффективно реконструировать направленный путь между любыми двумя связанными вершинами. Θ ( | Э | ) {\displaystyle \Тета (|E|)} Θ ( | В | ) {\displaystyle \Тета (|V|)}

Псевдокод

Массив prev[u][v]содержит предпоследнюю вершину на пути от uдо v(за исключением случая prev[v][v], где он всегда содержит , vдаже если нет замкнутого цикла на v): [11]

пусть dist будет массивом минимальных расстояний, инициализированным (бесконечностью) , пусть prev будет массивом индексов вершин, инициализированным значением null     |  В  |  ×  |  В  |    {\displaystyle |V|\times |V|}        {\displaystyle \infty}      |  В  |  ×  |  В  |    {\displaystyle |V|\times |V|} процедура  FloydWarshallWithPathReconstruction () для  каждого ребра (u, v) do dist[u][v] ← w(u, v) // Вес ребра (u, v) предыдущая[u][v] ← u для каждой вершины v сделать расст[v][v] ← 0 предыдущая[v][v] ← v для k от 1 до |V| do  // стандартная реализация Флойда-Уоршелла  для i от 1 до |V| для j от 1 до |V| if dist[i][j] > dist[i][k] + dist[k][j] then расстояние[i][j] ← расстояние[i][k] + расстояние[k][j] предыдущая[i][j] ← предыдущая[k][j]
Процедура  Path (u, v)  если prev[u][v] = null , то  возвращается [] путь ← [v] пока  uv  делать v ← предыдущая[u][v] путь.prepend(v) обратный путь

Сложность времени

Пусть будет , число вершин. Чтобы найти все ( для всех и ) из тех, что требуют операций. Поскольку мы начинаем с и вычисляем последовательность матриц , , , , каждая из которых имеет стоимость , общая временная сложность алгоритма составляет . н {\displaystyle n} | В | {\displaystyle |V|} н 2 {\displaystyle n^{2}} с час о г т е с т П а т час ( я , дж , к ) {\displaystyle \mathrm {shortestPath} (i,j,k)} я {\displaystyle я} дж {\displaystyle j} с час о г т е с т П а т час ( я , дж , к 1 ) {\displaystyle \mathrm {shortestPath} (i,j,k-1)} Θ ( н 2 ) {\displaystyle \Тета (n^{2})} с час о г т е с т П а т час ( я , дж , 0 ) = е г г е С о с т ( я , дж ) {\displaystyle \mathrm {shortestPath} (i,j,0)=\mathrm {edgeCost} (i,j)} н {\displaystyle n} с час о г т е с т П а т час ( я , дж , 1 ) {\displaystyle \mathrm {shortestPath} (i,j,1)} с час о г т е с т П а т час ( я , дж , 2 ) {\displaystyle \mathrm {shortestPath} (i,j,2)} {\displaystyle \ldots} с час о г т е с т П а т час ( я , дж , н ) {\displaystyle \mathrm {shortestPath} (i,j,n)} Θ ( н 2 ) {\displaystyle \Тета (n^{2})} н Θ ( н 2 ) = Θ ( н 3 ) {\displaystyle n\cdot \Тета (n^{2})=\Тета (n^{3})}

Приложения и обобщения

Алгоритм Флойда–Уоршелла можно использовать для решения, среди прочего, следующих задач:

Реализации

Реализации доступны для многих языков программирования .

  • Для C++ в библиотеке boost::graph
  • Для C# , на QuikGraph
  • Для C# — QuickGraphPCL (ответвление QuickGraph с лучшей совместимостью с проектами, использующими переносимые библиотеки классов).
  • Для Java в библиотеке Apache Commons Graph
  • Для JavaScript в библиотеке Cytoscape
  • Для Джулии в пакете Graphs.jl
  • Для MATLAB в пакете Matlab_bgl
  • Для Perl в модуле Graph
  • Для Python — в библиотеке SciPy (модуль scipy.sparse.csgraph) или библиотеке NetworkX
  • Для R в пакетах e1071 и Rfast

Сравнение с другими алгоритмами кратчайшего пути

Для графов с неотрицательными весами ребер алгоритм Дейкстры может быть использован для поиска всех кратчайших путей из одной вершины со временем выполнения . Таким образом, запуск Дейкстры, начиная с каждой вершины, занимает время . Поскольку , это дает наихудшее время выполнения повторного Дейкстры . Хотя это соответствует асимптотическому наихудшему времени выполнения алгоритма Флойда-Уоршелла, задействованные константы имеют довольно большое значение. Когда граф плотный (т. е. ), алгоритм Флойда-Уоршелла имеет тенденцию работать лучше на практике. Когда граф разрежен (т. е. значительно меньше ), Дейкстра имеет тенденцию доминировать. Θ ( | Э | + | В | бревно | В | ) {\displaystyle \Theta (|E|+|V|\log |V|)} Θ ( | Э | | В | + | В | 2 бревно | В | ) {\displaystyle \Theta (|E||V|+|V|^{2}\log |V|)} | Э | = О ( | В | 2 ) {\displaystyle |E|=O(|V|^{2})} О ( | В | 3 ) {\displaystyle O(|V|^{3})} | Э | | В | 2 {\displaystyle |E|\approx |V|^{2}} | Э | {\displaystyle |E|} | В | 2 {\displaystyle |V|^{2}}

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

Известны также алгоритмы, использующие быстрое умножение матриц для ускорения вычисления кратчайшего пути для всех пар вершин в плотных графах, но они обычно делают дополнительные предположения о весах ребер (например, требуют, чтобы они были небольшими целыми числами). [15] [16] Кроме того, из-за высоких постоянных множителей во времени их выполнения они обеспечат ускорение по сравнению с алгоритмом Флойда–Уоршелла только для очень больших графов.

Ссылки

  1. ^ аб Кормен, Томас Х .; Лейзерсон, Чарльз Э .; Ривест, Рональд Л. (1990). Введение в алгоритмы (1-е изд.). MIT Press и McGraw-Hill. ISBN 0-262-03141-8.См., в частности, раздел 26.2 «Алгоритм Флойда–Уоршелла», стр. 558–565 и раздел 26.4 «Общая структура для решения задач поиска путей в ориентированных графах», стр. 570–576.
  2. ^ Кеннет Х. Розен (2003). Дискретная математика и ее приложения, 5-е издание . Эддисон Уэсли. ISBN 978-0-07-119881-3.
  3. ^ Флойд, Роберт В. (июнь 1962 г.). «Алгоритм 97: Кратчайший путь». Сообщения ACM . 5 (6): 345. doi : 10.1145/367766.368168 . S2CID  2003382.
  4. ^ Рой, Бернард (1959). «Transitivité et connexité». ЧР акад. наук. Париж (на французском языке). 249 : 216–218.
  5. ^ Уоршалл, Стивен (январь 1962 г.). «Теорема о булевых матрицах». Журнал ACM . 9 (1): 11–12. doi : 10.1145/321105.321107 . S2CID  33763989.
  6. ^ Вайсштейн, Эрик В. «Алгоритм Флойда-Уоршелла». MathWorld .
  7. ^ Клини, SC (1956). «Представление событий в нервных сетях и конечных автоматах». В CE Shannon и J. McCarthy (ред.). Automata Studies . Princeton University Press. стр. 3–42.
  8. ^ Ингерман, Питер З. (ноябрь 1962 г.). «Алгоритм 141: Матрица путей». Сообщения ACM . 5 (11): 556. doi : 10.1145/368996.369016 . S2CID  29010500.
  9. ^ ab Hochbaum, Dorit (2014). "Раздел 8.9: Алгоритм Флойда-Уоршелла для всех пар кратчайших путей" (PDF) . Заметки к лекциям для IEOR 266: Графовые алгоритмы и сетевые потоки . Кафедра промышленной инженерии и исследования операций, Калифорнийский университет в Беркли . стр. 41–42.
  10. ^ Стефан Хугарди (апрель 2010 г.). «Алгоритм Флойда–Уоршелла на графах с отрицательными циклами». Information Processing Letters . 110 (8–9): 279–281. doi :10.1016/j.ipl.2010.02.001.
  11. ^ «Бесплатная книга алгоритмов».
  12. ^ Гросс, Джонатан Л.; Йеллен, Джей (2003), Справочник по теории графов, Дискретная математика и ее приложения, CRC Press, стр. 65, ISBN 9780203490204.
  13. ^ Пеналоза, Рафаэль. «Алгебраические структуры для транзитивного замыкания». CiteSeerX 10.1.1.71.7650 .  {{cite journal}}: Цитировать журнал требует |journal=( помощь )
  14. ^ Джиллис, Дональд (1993). Планирование задач с ограничениями предшествования И/ИЛИ (диссертация, Приложение B) (PDF) (Отчет).
  15. ^ Цвик, Ури (май 2002 г.), «Все пары кратчайших путей с использованием мостовых множеств и умножения прямоугольных матриц», Журнал ACM , 49 (3): 289–317, arXiv : cs/0008011 , doi : 10.1145/567112.567114, S2CID  1065901.
  16. ^ Чан, Тимоти М. (январь 2010 г.), «Больше алгоритмов для всех пар кратчайших путей во взвешенных графах», SIAM Journal on Computing , 39 (5): 2075–2089, CiteSeerX 10.1.1.153.6864 , doi :10.1137/08071990x .
  • Интерактивная анимация алгоритма Флойда–Уоршелла
  • Интерактивная анимация алгоритма Флойда–Уоршелла (Технический университет Мюнхена)
Взято с "https://en.wikipedia.org/w/index.php?title=Алгоритм_Флойда–Уоршалла&oldid=1250833503"