-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathSCC.cpp
110 lines (107 loc) · 2.7 KB
/
SCC.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
// Strongly Connected Components O(E + V)
// 二重辺,自己ループはOKだが,使う際に全体が連結かどうか注意!
class SCC {
private:
vector<vector<int>> &G;
vector<vector<int>> &rG;
vector<bool> used;
vector<int> vs;
int n;
void dfs(int v){
used[v] = true;
for(auto to : G[v]) if(!used[to]) dfs(to);
vs.pb(v);
}
void rdfs(int v, int k){
used[v] = true;
cmp[v] = k;
groups.back().pb(v);
for(auto to : rG[v]) if(!used[to]) rdfs(to, k);
}
public:
vector<int> cmp; //頂点iが属するSCCのID
vector<vector<int>> groups; // SCCのトポロジカル順i番目に属する頂点番号群
int gc = 0; // group count
SCC(vector<vector<int>> &g, vector<vector<int>> &r) : G(g), rG(r){
n = G.size();
used = vector<bool>(n, false);
cmp.resize(n);
rep(i,n) if(!used[i]) dfs(i);
fill(all(used), false);
for(int i=vs.size()-1; i>=0; i--) if(!used[vs[i]]){
groups.pb(vector<int>());
rdfs(vs[i], gc++);
}
}
inline bool same(int i, int j){ return cmp[i]==cmp[j]; }
inline int operator[](int i){ return cmp[i]; }
};
// 2-SAT O(M + N) M:# of vars, N:# of clauses
class TwoSat {
private:
vector<vector<int> > vec;
vector<vector<int> > rev;
vector<bool> res;
int v;
public:
TwoSat(int n) : v(n){
vec.resize(2*v);
rev.resize(2*v);
res.resize(v);
}
inline void add_edge(int a, bool ab, int b, bool bb){
// add (a -> b)
int va = a + (ab?0:v);
int vb = b + (bb?0:v);
vec[va].pb(vb);
rev[vb].pb(va);
}
inline void add(int a, bool ab, int b, bool bb){
// a or b <-> ((!a -> b) and (!b -> a))
add_edge(a, !ab, b, bb);
add_edge(b, !bb, a, ab);
}
bool exec(){
SCC scc(vec, rev);
rep(i,v){
if(scc.same(i, i+v)) return false;
res[i] = scc[i]>scc[i+v];
}
return true;
}
inline bool operator[](int i){ return res[i]; }
};
// for POJ
class SCC {
private:
public:
vector<vector<int> > &G;
vector<vector<int> > &rG;
vector<bool> used;
vector<int> vs;
vector<int> cmp;
int n;
int gc; // group count
void dfs(int v){
used[v] = true;
rep(i,G[v].size()) if(!used[G[v][i]]) dfs(G[v][i]);
vs.pb(v);
}
void rdfs(int v, int k){
used[v] = true;
cmp[v] = k;
rep(i,rG[v].size()) if(!used[rG[v][i]]) rdfs(rG[v][i], k);
}
SCC(vector<vector<int> > &g, vector<vector<int> > &r) : G(g), rG(r){
n = G.size();
used = vector<bool>(n,false);
vs.clear();
cmp.resize(n);
rep(i,n) if(!used[i]) dfs(i);
fill(all(used), false);
gc=0;
for(int i = vs.size()-1; i>=0; i--) if(!used[vs[i]]) rdfs(vs[i], gc++);
}
inline bool same(int i, int j){ return cmp[i]==cmp[j]; }
inline int operator[](int i){ return cmp[i]; }
};