Spaghetti Source logo

最大流 (Dinic)

目的

以下は 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;
}

Verified

PKU 1459, Power Network 924K, 281MS

前原 貴憲(maehara@prefield.com).

Last Modified: 2007.11.16 10:02:39.