以下は Dinic の層別ネットワークによって最大流問題を解くアルゴリズムである.
まず,始点から幅優先探索を行い,辺の本数の意味での最短路長 d を決定する.そして長さ d のフローを「流せる限り」流す.これを繰り返す.
「流せる限り」とは,そのフローを取り除くと増加パスがなくなることをいう.このようなフローのことをブロッキングフローというが,長さ d のブロッキングフローを流すと残余ネットワークの増加パスの長さが少なくとも 1 増加することが証明できる.そのため,高々 O(V) ステップでこのアルゴリズムは停止する.一回のステップは幅優先探索と O(V) 回の深さ優先探索で行えるので O(V E) である.したがって全体で O(V^2 E) で終了する.
O(V^2 E).
#define RESIDUE(s,t) (capacity[s][t]-flow[s][t]) Weight augment(const Graph &g, const Matrix &capacity, Matrix &flow, const vector<int> &level, vector<bool> &finished, int u, int t, Weight cur) { if (u == t || cur == 0) return cur; if (finished[u]) return 0; finished[u] = true; FOR(e, g[u]) if (level[e->dst] > level[u]) { Weight f = augment(g, capacity, flow, level, finished, e->dst, t, min(cur, RESIDUE(u, e->dst))); if (f > 0) { flow[u][e->dst] += f; flow[e->dst][u] -= f; finished[u] = false; return f; } } return 0; } Weight maximumFlow(const Graph &g, int s, int t) { int n = g.size(); Matrix flow(n, Array(n)), capacity(n, Array(n)); // adj. matrix REP(u,n) FOR(e,g[u]) capacity[e->src][e->dst] += e->weight; Weight total = 0; for (bool cont = true; cont; ) { cont = false; vector<int> level(n, -1); level[s] = 0; // make layered network queue<int> Q; Q.push(s); for (int d = n; !Q.empty() && level[Q.front()] < d; ) { int u = Q.front(); Q.pop(); if (u == t) d = level[u]; FOR(e, g[u]) if (RESIDUE(u,e->dst) > 0 && level[e->dst] == -1) Q.push(e->dst), level[e->dst] = level[u] + 1; } vector<bool> finished(n); // make blocking flows for (Weight f = 1; f > 0; ) { f = augment(g, capacity, flow, level, finished, s, t, INF); if (f == 0) break; total += f; cont = true; } } return total; }
PKU 1459, Power Network 924K, 281MS
Last Modified: 2007.11.16 10:02:39.