제출 #1081973

#제출 시각아이디문제언어결과실행 시간메모리
1081973RaresFelix카니발 티켓 (IOI20_tickets)C++17
100 / 100
700 ms88472 KiB
#include "tickets.h" #include <vector> #include <bits/stdc++.h> using namespace std; using ll = long long; using vi = vector<int>; using ii = pair<int, int>; ll find_maximum(int k, vector<vi> x) { int n = (int)x.size(); int m = (int)x[0].size(); // vector<vi> S = x; // for(int i = 0; i < n; ++i) // for(int j = 1; j < m; ++j) { // S[i][j] += S[i][j - 1]; // } vi A(n, 0); priority_queue<ii> PQ; auto calc = [&](int lin, int a) { return x[lin].end()[-a - 1] + x[lin][k - a - 1]; }; for(int i = 0; i < n; ++i) { PQ.push({calc(i, A[i]), i}); } ll sum = 0; for(int i = 0; i < n; ++i) { for(int j = 0; j < k; ++j) sum -= x[i][j]; } for(int i = 0; i < n * k / 2; ++i) { auto [val, id] = PQ.top(); PQ.pop(); sum += val; ++A[id]; if(A[id] < k) PQ.push({calc(id, A[id]), id}); } vector<vi> Re(n, vi(m, -1)); vector<vi> Poz(n), Neg(n); for(int i = 0; i < n; ++i) { for(int j = 0; j < k - A[i]; ++j) Neg[i].push_back(j); for(int j = 0; j < A[i]; ++j) Poz[i].push_back(m - 1 - j); } for(int nr = 0; nr < k; ++nr) { vi ord(n); iota(ord.begin(), ord.end(), 0); sort(ord.begin(), ord.end(), [&](auto a, auto b) { return min((int)Poz[a].size(), (int)Neg[a].size()) < min((int)Poz[b].size(), (int)Neg[b].size()); }); int nr_plus = n / 2, nr_minus = n / 2; auto assign = [&](int lin, int sgn) { int u; if(sgn == 1) { u = Poz[lin].back(); Poz[lin].pop_back(); --nr_plus; } else { u = Neg[lin].back(); Neg[lin].pop_back(); --nr_minus; } Re[lin][u] = nr; }; for(auto it : ord) { if(!nr_plus) { assign(it, -1); } else if(!nr_minus) { assign(it, 1); } else { if(Poz[it].size() > Neg[it].size()) assign(it, 1); else assign(it, -1); } } } allocate_tickets(Re); vi ideal(m, -1); iota(ideal.begin(), ideal.begin() + k, 0); sort(ideal.begin(), ideal.end()); ll s2 = 0; vector<vi> Seg(k); for(int i = 0; i < n; ++i) { for(int j = 0; j < m; ++j) if(Re[i][j] != -1) Seg[Re[i][j]].push_back(x[i][j]); sort(Re[i].begin(), Re[i].end()); assert(Re[i] == ideal); } for(int i = 0; i < k; ++i) { sort(Seg[i].begin(), Seg[i].end()); for(int j = 0; j < n / 2; ++j) s2 -= Seg[i][j]; for(int j = n / 2; j < n; ++j) s2 += Seg[i][j]; } assert(s2 == sum); return sum; }
#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...