Submission #854738

#TimeUsernameProblemLanguageResultExecution timeMemory
854738dlalswp25Circle selection (APIO18_circle_selection)C++17
49 / 100
3032 ms716456 KiB
#pragma GCC optimize("O3,unroll-loops") #pragma GCC target("avx2,bmi,bmi2,lzcnt,popcnt") #include <bits/stdc++.h> using namespace std; typedef pair<int, int> pii; const int K = 3; int N; int X[303030]; int Y[303030]; int R[303030]; int XL[303030]; int XR[303030]; pii UL[303030][22]; pii UR[303030][22]; int V[303030][44]; int ulp[303030]; int urp[303030]; int vp[303030]; int ans[303030]; char rr; struct DSU { int n; vector<int> l, r, c; vector<bool> chk; int cid; bool fg; void set(int _n) { n = _n; fg = false; } void init() { l.resize(n + 1); r.resize(n + 1); c.resize(n); chk.resize(n); cid = 0; l[0] = -1; r[0] = n; } inline int left(int x) { if (!fg) { fg = true; init(); } x = min(x, n - 1); return x < 0 ? -1 : (chk[x] ? x : l[c[x]]); } inline int right(int x) { if (!fg) { fg = true; init(); } x = max(x, 0); return x >= n ? n : (chk[x] ? x : r[c[x]]); } void upd(int x) { if (!fg) { fg = true; init(); } int lb = left(x), rb = right(x); chk[x] = true; cid++; if (x - lb <= rb - x) { for (int i = lb + 1; i < x; i++) c[i] = cid; l[cid] = lb; r[cid] = x; if (x < n - 1 && !chk[x + 1]) l[c[x + 1]] = x; } else { for (int i = x + 1; i < rb; i++) c[i] = cid; l[cid] = x; r[cid] = rb; if (x && !chk[x - 1]) r[c[x - 1]] = x; } } }; void rf(int &x) { x = 0; int sgn = 0; while (rr < 48 && rr != '-') rr = getchar(); if (rr == '-') { sgn = 1; rr = getchar(); } while (47 < rr) { x = (x << 3) + (x << 1) + (rr & 15); rr = getchar(); } if (sgn) x = -x; } inline bool intersect(int i, int j) { int dx = X[i] - X[j], dy = Y[i] - Y[j], r = R[i] + R[j]; return (long long)dx * dx + (long long)dy * dy <= (long long)r * r; } inline void upd_ans(int cur, int cand) { int &t = ans[cur]; if (R[t] > R[cand] || (R[t] == R[cand] && t < cand)) return; if (intersect(cur, cand)) t = cand; } struct SegTree { struct Node { vector<pii> v; DSU uf; }; int n, base; vector<Node> T; SegTree(int _n) : n(_n) { for (base = 1; base < n; base <<= 1); T.resize(base + base); } void add_line(int p, int q, int y, int i) { p += base; q += base; p--; q--; while (p <= q) { if (p & 1) { V[i][vp[i]++] = T[p].v.size(); T[p].v.emplace_back(y, i); p++; } if (~q & 1) { V[i][vp[i]++] = T[q].v.size(); T[q].v.emplace_back(y, i); q--; } p >>= 1; q >>= 1; } } void add_point(int p, int i, bool is_left) { p += base; p--; while (p) { if (is_left) UL[i][ulp[i]++] = {p, (int)T[p].v.size()}; else UR[i][urp[i]++] = {p, (int)T[p].v.size()}; p >>= 1; } } void init() { for (int i = 1; i < base + base; i++) T[i].uf.set(T[i].v.size()); } void upd(int p, int q, int i) { p += base; q += base; p--; q--; int ptr = 0; while (p <= q) { if (p & 1) { T[p].uf.upd(V[i][ptr++]); p++; } if (~q & 1) { T[q].uf.upd(V[i][ptr++]); q--; } p >>= 1; q >>= 1; } } void chk(int cur, bool is_left) { if (is_left) { for (int i = 0; i < ulp[cur]; i++) { int p, s; tie(p, s) = UL[cur][i]; int t = s; for (int j = 0; j < K; j++) { t = T[p].uf.right(t); if (t >= T[p].v.size()) break; upd_ans(cur, T[p].v[t].second); t++; } t = s; for (int j = 0; j < K; j++) { t = T[p].uf.left(t); if (t < 0) break; upd_ans(cur, T[p].v[t].second); t--; } } } else { for (int i = 0; i < urp[cur]; i++) { int p, s; tie(p, s) = UR[cur][i]; int t = s; for (int j = 0; j < K; j++) { t = T[p].uf.right(t); if (t >= T[p].v.size()) break; upd_ans(cur, T[p].v[t].second); t++; } t = s; for (int j = 0; j < K; j++) { t = T[p].uf.left(t); if (t < 0) break; upd_ans(cur, T[p].v[t].second); t--; } } } } }; int main() { rf(N); vector<int> v; for (int i = 1; i <= N; i++) { rf(X[i]); rf(Y[i]); rf(R[i]); XL[i] = X[i] - R[i]; XR[i] = X[i] + R[i]; v.push_back(XL[i]); v.push_back(XR[i]); } sort(v.begin(), v.end()); v.erase(unique(v.begin(), v.end()), v.end()); for (int i = 1; i <= N; i++) { XL[i] = lower_bound(v.begin(), v.end(), XL[i]) - v.begin() + 1; XR[i] = lower_bound(v.begin(), v.end(), XR[i]) - v.begin() + 1; } int M = v.size(); SegTree seg(M); vector<int> ord(N); iota(ord.begin(), ord.end(), 1); sort(ord.begin(), ord.end(), [&](int a, int b) { return Y[a] < Y[b]; }); for (int i : ord) { seg.add_line(XL[i], XR[i], Y[i], i); seg.add_point(XL[i], i, true); seg.add_point(XR[i], i, false); } seg.init(); sort(ord.begin(), ord.end(), [&](int a, int b) { if (R[a] == R[b]) return a < b; return R[a] > R[b]; }); for (int i : ord) { ans[i] = i; seg.chk(i, true); seg.chk(i, false); if (ans[i] == i) seg.upd(XL[i], XR[i], i); } for (int i = 1; i <= N; i++) printf("%d ", ans[i]); puts(""); return 0; }

Compilation message (stderr)

circle_selection.cpp: In member function 'void SegTree::chk(int, bool)':
circle_selection.cpp:161:12: warning: comparison of integer expressions of different signedness: 'int' and 'std::vector<std::pair<int, int> >::size_type' {aka 'long unsigned int'} [-Wsign-compare]
  161 |      if (t >= T[p].v.size()) break;
      |          ~~^~~~~~~~~~~~~~~~
circle_selection.cpp:179:12: warning: comparison of integer expressions of different signedness: 'int' and 'std::vector<std::pair<int, int> >::size_type' {aka 'long unsigned int'} [-Wsign-compare]
  179 |      if (t >= T[p].v.size()) break;
      |          ~~^~~~~~~~~~~~~~~~
circle_selection.cpp: In function 'int main()':
circle_selection.cpp:236:2: warning: this 'for' clause does not guard... [-Wmisleading-indentation]
  236 |  for (int i = 1; i <= N; i++) printf("%d ", ans[i]); puts("");
      |  ^~~
circle_selection.cpp:236:54: note: ...this statement, but the latter is misleadingly indented as if it were guarded by the 'for'
  236 |  for (int i = 1; i <= N; i++) printf("%d ", ans[i]); puts("");
      |                                                      ^~~~
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...