-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhopcroft-karp-bpm.cpp
111 lines (81 loc) · 2.26 KB
/
hopcroft-karp-bpm.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
#include <iostream>
#include <vector>
#include <queue>
using namespace std;
class HopCroftKarpBPM {
/*
U, V = size of two bipartite sets
Node IDs start from 1
*/
const int NIL = 0;
const int INF = 1<<30;
int U, V;
vector<int> Pair_U, Pair_V;
vector<vector<int> > Adj;
queue<int> Q;
vector<int> Dist;
bool BFS() {
Q = queue<int>();
for (int u = 1; u <= U; u ++) {
if (Pair_U[u] == NIL) {
Dist[u] = 0;
Q.push(u);
} else {
Dist[u] = INF;
}
}
Dist[NIL] = INF;
while (!Q.empty()) {
int u = Q.front();
Q.pop();
if (Dist[u] < Dist[NIL])
for (int v: Adj[u])
if (Dist[ Pair_V[v] ] == INF) {
Dist[ Pair_V[v] ] = Dist[u] + 1;
Q.push(Pair_V[v]);
}
}
return Dist[NIL] != INF;
}
bool DFS(int u) {
if (u != NIL) {
for (int v: Adj[u])
if (Dist[ Pair_V[v] ] == Dist[u] + 1)
if (DFS(Pair_V[v])) {
Pair_V[v] = u;
Pair_U[u] = v;
return true;
}
Dist[u] = INF;
return false;
}
return true;
}
public:
HopCroftKarpBPM() {}
HopCroftKarpBPM(int U, int V, const vector<vector<int> >& E) {
this->U = U;
this->V = V;
Pair_U = vector<int> (U + 1, NIL);
Pair_V = vector<int> (V + 1, NIL);
Dist = vector<int> (U + 1);
Adj = vector<vector<int> > (U + 1);
for (int i = 1; i <= U; i ++) {
Adj[i] = vector<int> (E[i].begin(), E[i].end());
}
}
int Solve() {
for (int u = 1; u <= U; u ++)
Pair_U[u] = NIL;
for (int v = 1; v <= V; v ++)
Pair_V[v] = NIL;
int matching = 0;
while (BFS()) {
for (int u = 1; u <= U; u ++)
if (Pair_U[u] == NIL)
if (DFS(u))
matching ++;
}
return matching;
}
};