最大フロー問題を解くアルゴリズムの1つ Ford-Fulkerson Algorithm
概要
最大流問題は、有向グラフ \(G = (V, E)\) の各辺 \(e \in E\) に容量 \(c(u, v)\) がついており、このグラフ上のsourceからsinkへ流せる流量を求める問題である。
最大フロー最小カット定理が成り立つため、最小カット問題も解くことができる。
Ford-Fulkerson Algorithm
sourceからsinkへのパスを片っ端から探し、そのパスに流せる限り流す。
計算量
流すフローを \(F\) とすると \(O(F|E|)\)
実装
# Ford-Fulkerson algorithm
class FordFulkerson:
def __init__(self, N):
self.N = N
self.G = [[] for i in range(N)]
def add_edge(self, fr, to, cap):
forward = [to, cap, None]
forward[2] = backward = [fr, 0, forward]
self.G[fr].append(forward)
self.G[to].append(backward)
def add_multi_edge(self, v1, v2, cap1, cap2):
edge1 = [v2, cap1, None]
edge1[2] = edge2 = [v1, cap2, edge1]
self.G[v1].append(edge1)
self.G[v2].append(edge2)
def dfs(self, v, t, f):
if v == t:
return f
used = self.used
used[v] = 1
for e in self.G[v]:
w, cap, rev = e
if cap and not used[w]:
d = self.dfs(w, t, min(f, cap))
if d:
e[1] -= d
rev[1] += d
return d
return 0
def flow(self, s, t):
flow = 0
f = INF = 10**9 + 7
N = self.N
while f:
self.used = [0]*N
f = self.dfs(s, t, INF)
flow += f
return flow
Verified
-
AOJ: "GRL_6_A: Network Flow - Maximum Flow": source (Python3, 0.75sec)