|
8 | 8 | #include <vector> |
9 | 9 |
|
10 | 10 | namespace weilycoder { |
11 | | -template <typename ptr_t = size_t> struct TwoEdgeConnectedComponents { |
12 | | - ptr_t dfs_time = 0, edge_time = 1; |
| 11 | +template <typename ptr_t = size_t> struct StrongConnectedComponents { |
| 12 | + ptr_t dfs_time = 0; |
13 | 13 |
|
14 | 14 | std::vector<bool> in_stack; |
15 | 15 | std::stack<ptr_t> stk; |
16 | 16 | std::vector<ptr_t> dfn, low; |
17 | | - std::vector<std::vector<std::pair<ptr_t, ptr_t>>> graph; |
| 17 | + std::vector<std::vector<ptr_t>> graph; |
18 | 18 |
|
19 | | - std::vector<std::vector<ptr_t>> eccs; |
| 19 | + std::vector<std::vector<ptr_t>> sccs; |
20 | 20 |
|
21 | | - TwoEdgeConnectedComponents(ptr_t n) |
| 21 | + StrongConnectedComponents(ptr_t n) |
22 | 22 | : in_stack(n, false), dfn(n, 0), low(n, 0), graph(n) {} |
23 | 23 |
|
24 | | - void add_edge(ptr_t u, ptr_t v) { |
25 | | - graph[u].emplace_back(v, edge_time); |
26 | | - graph[v].emplace_back(u, edge_time); |
27 | | - edge_time++; |
28 | | - } |
| 24 | + void add_edge(ptr_t u, ptr_t v) { graph[u].push_back(v); } |
29 | 25 |
|
30 | | - void tarjan(ptr_t u, ptr_t parent_edge) { |
| 26 | + void tarjan(ptr_t u) { |
31 | 27 | dfn[u] = low[u] = ++dfs_time; |
32 | 28 | stk.push(u), in_stack[u] = true; |
33 | 29 |
|
34 | | - for (const auto &[v, edge_id] : graph[u]) { |
35 | | - if (edge_id == parent_edge) |
36 | | - continue; |
| 30 | + for (const auto &v : graph[u]) { |
37 | 31 | if (!dfn[v]) |
38 | | - tarjan(v, edge_id), low[u] = std::min(low[u], low[v]); |
| 32 | + tarjan(v), low[u] = std::min(low[u], low[v]); |
39 | 33 | else if (in_stack[v]) |
40 | 34 | low[u] = std::min(low[u], dfn[v]); |
41 | 35 | } |
42 | 36 |
|
43 | 37 | if (dfn[u] == low[u]) { |
44 | | - eccs.emplace_back(); |
45 | | - eccs.back().push_back(u); |
| 38 | + sccs.emplace_back(); |
| 39 | + sccs.back().push_back(u); |
46 | 40 | in_stack[u] = false; |
47 | 41 | while (stk.top() != u) |
48 | | - eccs.back().push_back(stk.top()), in_stack[stk.top()] = false, stk.pop(); |
| 42 | + sccs.back().push_back(stk.top()), in_stack[stk.top()] = false, stk.pop(); |
49 | 43 | stk.pop(); |
50 | 44 | } |
51 | 45 | } |
52 | 46 |
|
53 | 47 | void solve() { |
54 | 48 | for (size_t i = 0; i < graph.size(); ++i) |
55 | 49 | if (!dfn[i]) |
56 | | - tarjan(i, 0); |
| 50 | + tarjan(i); |
| 51 | + std::reverse(sccs.begin(), sccs.end()); |
57 | 52 | } |
58 | 53 | }; |
59 | 54 |
|
60 | | -template <typename ptr_t = size_t> struct StrongConnectedComponents { |
61 | | - ptr_t dfs_time = 0; |
| 55 | +template <typename ptr_t = size_t> struct TwoEdgeConnectedComponents { |
| 56 | + ptr_t dfs_time = 0, edge_time = 1; |
62 | 57 |
|
63 | 58 | std::vector<bool> in_stack; |
64 | 59 | std::stack<ptr_t> stk; |
65 | 60 | std::vector<ptr_t> dfn, low; |
66 | | - std::vector<std::vector<ptr_t>> graph; |
| 61 | + std::vector<std::vector<std::pair<ptr_t, ptr_t>>> graph; |
67 | 62 |
|
68 | | - std::vector<std::vector<ptr_t>> sccs; |
| 63 | + std::vector<std::vector<ptr_t>> eccs; |
69 | 64 |
|
70 | | - StrongConnectedComponents(ptr_t n) |
| 65 | + TwoEdgeConnectedComponents(ptr_t n) |
71 | 66 | : in_stack(n, false), dfn(n, 0), low(n, 0), graph(n) {} |
72 | 67 |
|
73 | | - void add_edge(ptr_t u, ptr_t v) { graph[u].push_back(v); } |
| 68 | + void add_edge(ptr_t u, ptr_t v) { |
| 69 | + graph[u].emplace_back(v, edge_time); |
| 70 | + graph[v].emplace_back(u, edge_time); |
| 71 | + edge_time++; |
| 72 | + } |
74 | 73 |
|
75 | | - void tarjan(ptr_t u) { |
| 74 | + void tarjan(ptr_t u, ptr_t parent_edge) { |
76 | 75 | dfn[u] = low[u] = ++dfs_time; |
77 | 76 | stk.push(u), in_stack[u] = true; |
78 | 77 |
|
79 | | - for (const auto &v : graph[u]) { |
| 78 | + for (const auto &[v, edge_id] : graph[u]) { |
| 79 | + if (edge_id == parent_edge) |
| 80 | + continue; |
80 | 81 | if (!dfn[v]) |
81 | | - tarjan(v), low[u] = std::min(low[u], low[v]); |
| 82 | + tarjan(v, edge_id), low[u] = std::min(low[u], low[v]); |
82 | 83 | else if (in_stack[v]) |
83 | 84 | low[u] = std::min(low[u], dfn[v]); |
84 | 85 | } |
85 | 86 |
|
86 | 87 | if (dfn[u] == low[u]) { |
87 | | - sccs.emplace_back(); |
88 | | - sccs.back().push_back(u); |
| 88 | + eccs.emplace_back(); |
| 89 | + eccs.back().push_back(u); |
89 | 90 | in_stack[u] = false; |
90 | 91 | while (stk.top() != u) |
91 | | - sccs.back().push_back(stk.top()), in_stack[stk.top()] = false, stk.pop(); |
| 92 | + eccs.back().push_back(stk.top()), in_stack[stk.top()] = false, stk.pop(); |
92 | 93 | stk.pop(); |
93 | 94 | } |
94 | 95 | } |
95 | 96 |
|
96 | 97 | void solve() { |
97 | 98 | for (size_t i = 0; i < graph.size(); ++i) |
98 | 99 | if (!dfn[i]) |
99 | | - tarjan(i); |
100 | | - std::reverse(sccs.begin(), sccs.end()); |
| 100 | + tarjan(i, 0); |
101 | 101 | } |
102 | 102 | }; |
103 | 103 |
|
|
0 commit comments