이 제출은 이전 버전의 oj.uz에서 채점하였습니다. 현재는 제출 당시와는 다른 서버에서 채점을 하기 때문에, 다시 제출하면 결과가 달라질 수도 있습니다.
#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 time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |
# | Verdict | Execution time | Memory | Grader output |
---|
Fetching results... |