제출 #839707

#제출 시각아이디문제언어결과실행 시간메모리
839707model_code봉쇄 시간 (IOI23_closing)C++17
100 / 100
309 ms48816 KiB
// correct/solution-wiwitrifai.cpp #include "closing.h" #include <bits/stdc++.h> using namespace std; constexpr long long kInf = 2e18; template <typename T> using min_priority_queue = priority_queue<T, vector<T>, greater<T>>; struct GreedyPicker { vector<long long> vals; priority_queue<pair<long long, int>> ins; min_priority_queue<pair<long long, int>> outs; vector<bool> is_ins, is_outs, active; long long total; int cnt; GreedyPicker(vector<long long> _vals, bool _active) : vals(_vals) { int n = vals.size(); is_ins.assign(n, false); is_outs.assign(n, _active); active.assign(n, _active); cnt = 0; total = 0; if (_active) { for (int i = 0; i < n; ++i) { outs.emplace(vals[i], i); } } } bool inc() { while (!outs.empty()) { auto [val, id] = outs.top(); outs.pop(); is_outs[id] = false; if (!active[id]) continue; total += val; ++cnt; ins.emplace(val, id); is_ins[id] = true; return true; } return false; } bool dec() { while (!ins.empty()) { auto [val, id] = ins.top(); ins.pop(); is_ins[id] = false; if (!active[id]) continue; total -= val; --cnt; outs.emplace(val, id); is_outs[id] = true; return true; } return false; } long long top() { while (!ins.empty() && !active[ins.top().second]) ins.pop(); return ins.empty() ? 0 : ins.top().first; } long long check_inc() { while (!outs.empty() && !active[outs.top().second]) outs.pop(); return outs.empty() ? kInf : outs.top().first; } void deactivate(int v) { if (!active[v]) return; if (is_ins[v]) { --cnt; total -= vals[v]; } active[v] = false; } void activate(int v) { if (active[v]) return; active[v] = true; if (is_ins[v]) { ++cnt; total += vals[v]; } else if (!is_outs[v]) { if (vals[v] <= top()) { ins.emplace(vals[v], v); ++cnt; total += vals[v]; } else { outs.emplace(vals[v], v); } } } }; vector<vector<pair<int, int>>> tree; void dfs(int v, int par, vector<long long> &dist) { for (auto [u, w] : tree[v]) { if (u == par) continue; dist[u] = dist[v] + w; dfs(u, v, dist); } } int max_score(int N, int X, int Y, long long K, vector<int> U, vector<int> V, vector<int> W) { tree.assign(N, {}); for (int i = 0; i < N - 1; ++i) { tree[U[i]].emplace_back(V[i], W[i]); tree[V[i]].emplace_back(U[i], W[i]); } vector<long long> from_x(N, 0), from_y(N, 0); dfs(X, X, from_x); dfs(Y, Y, from_y); vector<long long> val1(N), val2(N), val_diff(N); for (int i = 0; i < N; ++i) { val1[i] = min(from_x[i], from_y[i]); val2[i] = max(from_x[i], from_y[i]); val_diff[i] = val2[i] - val1[i]; } vector<int> sorted(N); iota(sorted.begin(), sorted.end(), 0); int ans = 0; { sort(sorted.begin(), sorted.end(), [&](int l, int r) { return val1[l] < val1[r]; }); long long total = 0; for (auto v : sorted) { if (total + val1[v] > K) break; ++ans; total += val1[v]; } } GreedyPicker picker_one(val1, true), picker_two(val2, false); auto in_path = [&](int v) { return from_x[v] + from_y[v] == from_x[Y]; }; int cur = 0; for (int v = 0; v < N; ++v) { if (in_path(v)) { K -= val1[v]; ++cur; picker_one.deactivate(v); } } sort(sorted.begin(), sorted.end(), [&](int l, int r) { return make_pair(val_diff[l], val1[l]) < make_pair(val_diff[r], val1[r]); }); for (auto v : sorted) { if (!in_path(v)) { picker_two.activate(v); picker_one.deactivate(v); } else { K -= val_diff[v]; ++cur; } if (K < 0) break; // make sure that the total cost <= K while (picker_one.total + picker_two.total > K) { if (picker_two.top() < 2 * picker_one.top()) { picker_one.dec(); } else { picker_two.dec(); } } // balance between picker_one.top() and picker_two.top()/2 while (2 * picker_one.top() > picker_two.check_inc()) { if (picker_one.total + picker_two.total + picker_two.check_inc() <= K) { picker_two.inc(); } else { picker_one.dec(); } } while (2 * picker_one.check_inc() < picker_two.top()) { if (picker_one.total + picker_two.total + picker_one.check_inc() <= K) { picker_one.inc(); } else { picker_two.dec(); } } // pick more to fill the remaining space while (true) { auto inc_one = picker_one.check_inc(); auto inc_two = picker_two.check_inc(); auto remaining = K - picker_one.total - picker_two.total; if (min(inc_one, inc_two) > remaining) break; if (inc_one > remaining || (inc_two <= remaining && inc_two <= inc_one * 2)) picker_two.inc(); else picker_one.inc(); } ans = max(ans, cur + picker_two.cnt * 2 + picker_one.cnt); // try to pick two by sacrificing one if it's possible picker_two.inc(); picker_one.dec(); if (picker_one.total + picker_two.total <= K) { ans = max(ans, cur + picker_two.cnt * 2 + picker_one.cnt); } } return ans; }
#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...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...