Алгоритм Форда-Фалкерсона для поиска максимального паросочетания
НЕТ ВОЙНЕ |
24 февраля 2022 года российское руководство во главе с Владимиром Путиным развязало агрессивную войну против Украины. В глазах всего мира это военное преступление совершено от лица всей страны, всех россиян. Будучи гражданами Российской Федерации, мы против своей воли оказались ответственными за нарушение международного права, военное вторжение и массовую гибель людей. Чудовищность совершенного преступления не оставляет возможности промолчать или ограничиться пассивным несогласием. Мы убеждены в абсолютной ценности человеческой жизни, в незыблемости прав и свобод личности. Режим Путина — угроза этим ценностям. Наша задача — обьединить все силы для сопротивления ей. Эту войну начали не россияне, а обезумевший диктатор. И наш гражданский долг — сделать всё, чтобы её остановить. Антивоенный комитет России |
Распространяйте правду о текущих событиях, оберегайте от пропаганды своих друзей и близких. Изменение общественного восприятия войны - ключ к её завершению. |
meduza.io, Популярная политика, Новая газета, zona.media, Майкл Наки. |
Содержание
Идея алгоритма
Пусть дан неориентированный двудольный граф и требуется найти максимальное паросочетание в нём. Обозначим доли исходного графа как и . Построим граф следующим образом:
(т.е. добавим новый исток и сток );
.
Изначально текущее паросочетание пусто. На каждом шаге алгоритма будем поддерживать следующий инвариант: в текущее найденное паросочетание входят те и только те ребра, которые направлены из
в .- Ищем в графе поиском в глубину. путь из в
- Если путь найден, перезаписываем текущее паросочетание. Далее инвертируем все рёбра на пути (ребро становится ребром ) и удаляем и ребра, покрывающие вершины, принадлежащие текущему паросочетанию.
- Если путь не был найден, значит текущее паросочетание является максимальным, и алгоритм завершает работу. Иначе переходим к пункту 1.
Корректность алгоритма
Обозначим как теоремы: если мы на каком-то шаге можем найти новый путь, т.е дополняющую цепь, то мы увеличиваем текущее паросочетание. Если путь найти мы уже не можем, значит дополняющих цепей в графе нет и текущее паросочетание — искомое. Осталось доказать что сделанное предположение действительно верно.
путь из в без первого и последнего ребра. Пусть он является дополняющей цепью для исходного графа , и пусть также существование дополняющей цепи в графе приводит к существованию пути . Тогда изТ. к.
— путь в двудольном графе, начинающийся в и заканчивающийся в , то он нечетной длины. Вершины в нем не повторяются (т.к. это путь в дереве поиска в глубину). Рассмотрим текущее паросочетание. Согласно поддерживаемому инварианту -ребра в паросочетании, а -ребра — нет. В таком случае ребра пути можно пронумеровать так, чтобы нечетные ребра были свободными, а четные — покрытыми ребрами текущего паросочетания. Заметим, что путь может начинаться и заканчиваться только в свободной вершине, т. к. из ведут ребра только в свободные вершины и только из свободных вершин ведут ребра в . Итак, теперь ясно, что — дополняющая цепь для графа .Обратно, пусть существует дополняющая цепь в графе
. В одной из ориентаций она начинается в какой-то свободной вершине и заканчивается в свободной вершине , далее будем рассматривать именно эту ориентацию. Ребра поочередно то не лежат, то лежат в паросочетании, значит в нашей ориентации эти ребра поочередно ориентированы то , то . Заметим что эта ориентация совпадает с ориентацией ребер на пути, а значит в нашем ориентированом графе существует путь из свободной вершины в свободную вершину . Нo каждая свободная вершина из связана ребром с в графе , аналогично каждая свободная вершина из связана ребром с . Не сложно заметить, что, в таком случае, достижим из , а значит в процессе поиска в глубину будет найден некий путь и соответствующий ему .Утверждение доказано.
Оценка производительности
Поиск в глубину запускается от вершины
не более чем раз, т.к. из ведет ровно ребер, и при каждом запуске одно из них инвертируется. Сам поиск работает за , каждая инвертация и перезапись паросочетания так же занимает времени. Тогда все время алгоритма ограничено .Псевдокод
- — массив вершин , инцидентные в текущем паросочетании,
- — массив вершин , инцидентные в текущем паросочетании,
- — массив, где помечаются посещенные вершины.
Максимальное паросочетание — такие ребра
, что .Поиск в глубину, одновременно инвертирующий ребра:
bool dfs(x):
if vis[x]
return false
vis[x] = true
for
if py[y] == -1
py[y] = x
px[x] = y
return true
else
if dfs(py[y])
py[y] = x
px[x] = y
return true
return false
Инициализация и внешний цикл:
func fordFulkerson():
fill(px, -1)
fill(py, -1)
isPath = true
while isPath
isPath = false
fill(vis, false)
for
if px[x] == -1
if dfs(x)
isPath = true
См. также
- Теорема о максимальном паросочетании и дополняющих цепях
- Алгоритм Форда-Фалкерсона, реализация с помощью поиска в глубину
Источники информации
- Томас Х. Кормен, Чарльз И. Лейзерсон, Рональд Л. Ривест, Клиффорд Штайн — "Алгоритмы: построение и анализ", 2-е издание, стр. 758 - 761.