Алгоритм Форда-Фалкерсона для поиска максимального паросочетания
Идея алгоритма
Пусть дан неориентированный двудольный граф максимальное паросочетание в нём. Обозначим доли исходного графа как
и требуется найтии . Построим граф следующим образом:
(т.е. добавим новый исток и сток );
.
Изначально максимальное паросочетание пусто. На каждом шаге алгоритма в него входят ребра, ведущие из
в .- Ищем в графе путь из в поиском в глубину.
- Если путь найден, перезаписываем текущее паросочетание. Теперь инвертируем все рёбра на пути (ребро становится ребром ) и удаляем и ребра, покрывающие вершины, принадлежащие текущему паросочетанию.
- Если путь не был найден, значит текущее паросочетание является максимальным, и алгоритм завершает работу. Иначе переходим к пункту 1.
Корректность алгоритма
Обозначим как теоремы - если мы на каждом шаге можем найти новый путь, т.е. находим новую дополняющую цепь, то мы увеличиваем текущее паросочетание. Если путь найти мы уже не можем, то текущее паросочетание - искомое. Осталось доказать, что путь действительно является дополняющей цепью.
путь из в без первого и последнего ребра. Пусть он является дополняющей цепью для исходного графа . Тогда изТ.к.
- путь в двудольном графе, он нечетной длины, в котором вершины не повторяются (т.к. этот путь строится с помощью поиска в глубину). В таком случае его ребра можно пронумеровать так, чтобы нечетные ребра были свободными, а четные - покрытыми, т.е. только ребра из в включаем в паросочетание(что соответствует инварианту его построения). Тогда этот путь - дополняющая цепь для графа , алгоритм корректен.Оценка производительности
Поиск в глубину запускается от вершины
не более чем раз, т.к. из ведет ровно ребер, и при каждом запуске одно из них инвертируется. Сам поиск работает за , каждая инвертация и перезапись паросочетания так же занимает времени. Тогда все время алгоритма ограничено .Псевдокод
В массиве
хранятся вершины , инцидентные в текущем паросочетании, для аналогично. Максимальное паросочетание - такие ребра , что . 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
px[] = -1
py[] = -1
is_path = true;
while (is_path)
is_path = false
vis[] = false
for
if (px[x] == -1)
if dfs(x)
is_path = true