-
Notifications
You must be signed in to change notification settings - Fork 12
/
Copy pathflow_with_lower_bound.cpp
121 lines (101 loc) · 2.62 KB
/
flow_with_lower_bound.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
/*
Flow with lower bound
Tested: ZOJ 3229
Complexity: O(n^2 m)
*/
template <typename T> struct dinic {
struct edge {
int src, dst;
T low, cap, flow;
int rev;
};
int n;
vector<vector<edge>> adj;
dinic(int n) : n(n), adj(n + 2) {}
void add_edge(int src, int dst, T low, T cap) {
adj[src].push_back({src, dst, low, cap, 0, (int)adj[dst].size()});
if (src == dst)
adj[src].back().rev++;
adj[dst].push_back({dst, src, 0, 0, 0, (int)adj[src].size() - 1});
}
vector<int> level, iter;
T augment(int u, int t, T cur) {
if (u == t)
return cur;
for (int &i = iter[u]; i < (int)adj[u].size(); ++i) {
edge &e = adj[u][i];
if (e.cap - e.flow > 0 && level[u] > level[e.dst]) {
T f = augment(e.dst, t, min(cur, e.cap - e.flow));
if (f > 0) {
e.flow += f;
adj[e.dst][e.rev].flow -= f;
return f;
}
}
}
return 0;
}
int bfs(int s, int t) {
level.assign(n + 2, n + 2);
level[t] = 0;
queue<int> Q;
for (Q.push(t); !Q.empty(); Q.pop()) {
int u = Q.front();
if (u == s)
break;
for (edge &e : adj[u]) {
edge &erev = adj[e.dst][e.rev];
if (erev.cap - erev.flow > 0 && level[e.dst] > level[u] + 1) {
Q.push(e.dst);
level[e.dst] = level[u] + 1;
}
}
}
return level[s];
}
const T oo = numeric_limits<T>::max();
T max_flow(int source, int sink) {
vector<T> delta(n + 2);
for (int u = 0; u < n; ++u) // initialize
for (auto &e : adj[u]) {
delta[e.src] -= e.low;
delta[e.dst] += e.low;
e.cap -= e.low;
e.flow = 0;
}
T sum = 0;
int s = n, t = n + 1;
for (int u = 0; u < n; ++u) {
if (delta[u] > 0) {
add_edge(s, u, 0, delta[u]);
sum += delta[u];
} else if (delta[u] < 0)
add_edge(u, t, 0, -delta[u]);
}
add_edge(sink, source, 0, oo);
T flow = 0;
while (bfs(s, t) < n + 2) {
iter.assign(n + 2, 0);
for (T f; (f = augment(s, t, oo)) > 0;)
flow += f;
}
if (flow != sum)
return -1; // no solution
for (int u = 0; u < n; ++u)
for (auto &e : adj[u]) {
e.cap += e.low;
e.flow += e.low;
edge &erev = adj[e.dst][e.rev];
erev.cap -= e.low;
erev.flow -= e.low;
}
adj[sink].pop_back();
adj[source].pop_back();
while (bfs(source, sink) < n + 2) {
iter.assign(n + 2, 0);
for (T f; (f = augment(source, sink, oo)) > 0;)
flow += f;
} // level[u] == n + 2 ==> s-side
return flow;
}
};