forked from Ashishgup1/Competitive-Coding
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathMaxFlow - Push Relabel [V^2 sqrt(E)].cpp
113 lines (100 loc) · 2.26 KB
/
MaxFlow - Push Relabel [V^2 sqrt(E)].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
//Push-Relabel Algorithm for Flows, Complexity: O(V^2 √E)
//To obtain the actual flow values, look at all edges with capacity > 0
//Zero capacity edges are residual edges
struct edge
{
int from, to, cap, flow, index;
edge(int from, int to, int cap, int flow, int index):
from(from), to(to), cap(cap), flow(flow), index(index) {}
};
struct PushRelabel
{
static const long long INF=1e18;
int n;
vector<vector<edge> > g;
vector<long long> excess;
vector<int> height;
PushRelabel(int n):
n(n), g(n), excess(n), height(n) {}
void addEdge(int from, int to, int cap)
{
g[from].push_back(edge(from, to, cap, 0, g[to].size()));
if(from==to)
g[from].back().index++;
g[to].push_back(edge(to, from, 0, 0, g[from].size()-1));
}
void push(edge &e)
{
int amt=(int)min(excess[e.from], (long long)e.cap - e.flow);
if(height[e.from]<=height[e.to] || amt==0)
return;
e.flow += amt;
g[e.to][e.index].flow -= amt;
excess[e.to] += amt;
excess[e.from] -= amt;
}
void relabel(int u)
{
int d=2e5;
for(auto &it:g[u])
{
if(it.cap-it.flow>0)
d=min(d, height[it.to]);
}
if(d<INF)
height[u]=d+1;
}
vector<int> find_max_height_vertices(int source, int dest)
{
vector<int> max_height;
for(int i=0;i<n;i++)
{
if(i!=source && i!=dest && excess[i]>0)
{
if(!max_height.empty() && height[i] > height[max_height[0]])
max_height.clear();
if(max_height.empty() || height[i] == height[max_height[0]])
max_height.push_back(i);
}
}
return max_height;
}
long long max_flow(int source, int dest)
{
excess.assign(n, 0);
height.assign(n, 0);
height[source]=n;
excess[source]=INF;
for(auto &it:g[source])
push(it);
vector<int> current;
while(!(current = find_max_height_vertices(source, dest)).empty())
{
for(auto i:current)
{
bool pushed=false;
for(auto &e:g[i])
{
if(excess[i]==0)
break;
if(e.cap - e.flow>0 && height[e.from] == height[e.to] + 1)
{
push(e);
pushed=true;
}
}
if(!pushed)
{
relabel(i);
break;
}
}
}
long long max_flow=0;
for(auto &e:g[source])
max_flow+=e.flow;
return max_flow;
}
};
//Problem 1: http://codeforces.com/contest/546/problem/E
//Solution 1: http://codeforces.com/contest/546/submission/40528334