Submission #219243

#TimeUsernameProblemLanguageResultExecution timeMemory
219243IgorIWerewolf (IOI18_werewolf)C++17
0 / 100
1353 ms132576 KiB
#include <iostream> #include <map> #include <vector> #include <algorithm> using namespace std; typedef vector<int> vi; const int N = 3e5 + 55; int n, m, q; vi graph[N]; int sz1[N]; int root1[N]; int mx1[N]; vi tree1[N]; int sz2[N]; int root2[N]; int mn2[N]; vi tree2[N]; int Root1(int x) { if (root1[x] == x) return x; return root1[x] = Root1(root1[x]); } int Root2(int x) { if (root2[x] == x) return x; return root2[x] = Root2(root2[x]); } void Merge1(int v, int u) { v = Root1(v); u = Root1(u); int x = mx1[v], y = mx1[u]; if (v != u) { tree1[x].push_back(y); tree1[y].push_back(x); if (sz1[v] < sz1[u]) { sz1[u] += sz1[v]; root1[v] = u; mx1[u] = max(x, y); } else { sz1[v] += sz1[u]; root1[u] = v; mx1[v] = max(x, y); } } } void Merge2(int v, int u) { v = Root2(v); u = Root2(u); int x = mn2[v], y = mn2[u]; if (v != u) { tree2[x].push_back(y); tree2[y].push_back(x); if (sz2[v] < sz2[u]) { sz2[u] += sz2[v]; root2[v] = u; mn2[u] = min(x, y); } else { sz2[v] += sz2[u]; root2[u] = v; mn2[v] = min(x, y); } } } int up1[20][N]; int up2[20][N]; void dfs1(int v, int p) { up1[0][v] = p; for (auto u : tree1[v]) if (u != p) { dfs1(u, v); } } void dfs2(int v, int p) { up2[0][v] = p; for (auto u : tree2[v]) if (u != p) { dfs2(u, v); } } int id[N], t = 0; void dfs3(int v, int p) { id[v] = t++; for (auto u : tree2[v]) if (u != p) { dfs3(u, v); } } vi tree3[N], tree4[N]; int L3[N], R3[N]; int L4[N], R4[N]; vector<int> euler; bool comp(pair<pair<int, int>, int> a, pair<pair<int, int>, int> b) { return R3[a.first.first] < R3[b.first.first]; } void dfs4(int v, int p) { L4[v] = v; R4[v] = v; for (auto u : tree4[v]) if (u != p) { dfs4(u, v); R4[v] = R4[u]; } } void dfs5(int v, int p) { L3[v] = euler.size(); R3[v] = euler.size(); euler.push_back(v); for (auto u : tree3[v]) if (u != p) { dfs5(u, v); R3[v] = euler.size(); euler.push_back(v); } } int ar[N]; void _set(int p, int v) { ar[p] = v; } int get_max(int l, int r) { int res = 0; for (int i = l; i <= r; i++) res = max(res, ar[i]); return res; } vector<int> check_validity(int n0, vector<int> x, vector<int> y, vector<int> s, vector<int> e, vector<int> l, vector<int> r) { n = n0; m = x.size(); q = s.size(); for (int i = 0; i < m; i++) { graph[x[i]].push_back(y[i]); graph[y[i]].push_back(x[i]); } vi ans(q); fill(sz1, sz1 + n, 1); fill(sz2, sz2 + n, 1); for (int i = 0; i < n; i++) root1[i] = i, root2[i] = i, mx1[i] = i, mn2[i] = i; for (int i = 0; i < n; i++) { for (auto u : graph[i]) { if (u < i) Merge1(u, i); } } for (int i = n - 1; i >= 0; i--) { for (auto u : graph[i]) { if (u > i) Merge2(u, i); } } dfs1(n - 1, n - 1); dfs2(0, 0); for (int j = 1; j < 20; j++) { for (int i = 0; i < n; i++) { int x = up1[j - 1][i]; up1[j][i] = up1[j - 1][x]; int y = up2[j - 1][i]; up2[j][i] = up2[j - 1][y]; } } vector<pair<pair<int, int>, int> > ques; for (int i = 0; i < q; i++) { int st = s[i]; for (int j = 19; j >= 0; j--) { if (l[i] <= up2[j][st]) st = up2[j][st]; } int fn = e[i]; for (int j = 19; j >= 0; j--) { if (up1[j][fn] <= r[i]) fn = up1[j][fn]; } ques.push_back({{fn, st}, i}); } dfs3(0, 0); for (int i = 0; i < n; i++) { for (auto u : tree1[i]) { int x = id[i], y = id[u]; tree3[x].push_back(y); } for (auto u : tree2[i]) { int x = id[i], y = id[u]; tree4[x].push_back(y); } } for (int i = 0; i < ques.size(); i++) { ques[i].first.first = id[ques[i].first.first]; ques[i].first.second = id[ques[i].first.second]; } dfs4(0, 0); dfs5(id[n - 1], id[n - 1]); sort(ques.begin(), ques.end(), comp); /*for (int i = 0; i < q; i++) { cout << " ! " << ques[i].first.first << " " << ques[i].first.second << " " << ques[i].second << "\n"; } for (int i = 0; i < n; i++) { cout << L3[i] << " " << R3[i] << "\n"; } cout << endl; for (int i = 0; i < n; i++) { cout << L4[i] << " " << R4[i] << "\n"; } cout << endl;*/ int j = 0; for (int r0 = 0; r0 < euler.size(); r0++) { int v = euler[r0]; //cout << "watch " << v << endl; int l0 = L3[v]; for (; j < ques.size();) { if (r0 == R3[v] && ques[j].first.first == v) { //cout << "answer query " << ques[j].second << endl; int d = get_max(L4[ques[j].first.second], R4[ques[j].first.second]); if (l0 <= d) { ans[ques[j].second] = 1; } j++; } else break; } _set(v, r0); } return ans; } /*int main() { int n, m, q; cin >> n >> m >> q; vector<int> x(m), y(m); for (int i = 0; i < m; i++) cin >> x[i] >> y[i]; vector<int> s(q), e(q), l(q), r(q); for (int i = 0; i < q; i++) cin >> s[i] >> e[i] >> l[i] >> r[i]; vector<int> ans = check_validity(n, x, y, s, e, l, r); for (int i = 0; i < q; i++) cout << ans[i] << endl; }*/

Compilation message (stderr)

werewolf.cpp: In function 'std::vector<int> check_validity(int, std::vector<int>, std::vector<int>, std::vector<int>, std::vector<int>, std::vector<int>, std::vector<int>)':
werewolf.cpp:239:23: warning: comparison between signed and unsigned integer expressions [-Wsign-compare]
     for (int i = 0; i < ques.size(); i++)
                     ~~^~~~~~~~~~~~~
werewolf.cpp:262:25: warning: comparison between signed and unsigned integer expressions [-Wsign-compare]
     for (int r0 = 0; r0 < euler.size(); r0++)
                      ~~~^~~~~~~~~~~~~~
werewolf.cpp:267:18: warning: comparison between signed and unsigned integer expressions [-Wsign-compare]
         for (; j < ques.size();)
                ~~^~~~~~~~~~~~~
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...