This submission is migrated from previous version of oj.uz, which used different machine for grading. This submission may have different result if resubmitted.
#include "candies.h"
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
mt19937 rd(179);
struct node {
node *l, *r;
int x, y;
int val;
int sz;
int sum;
int max_suf;
int min_suf;
node(int x, int val) : x(x), val(val) {
l = r = NULL;
y = rd();
sz = 1;
sum = val;
max_suf = max(0, val);
min_suf = min(0, val);
}
};
int get_sz(node *n) { return (n == NULL) ? 0 : n->sz; }
int get_sum(node *n) { return (n == NULL) ? 0 : n->sum; }
int get_max_suf(node *n) { return (n == NULL) ? 0 : n->max_suf; }
int get_min_suf(node *n) { return (n == NULL) ? 0 : n->min_suf; }
int get_ct(node *n, int i) {
if (i == get_sz(n->l)) {
return n->val;
}
if (i < get_sz(n->l)) {
return get_ct(n->l, i);
}
return get_ct(n->r, i - get_sz(n->l) - 1);
}
int getInd(node *n, int x) {
if (n == NULL)
return 0;
if (x < n->x)
return getInd(n->l, x);
if (x == n->x)
return get_sz(n->l);
return getInd(n->r, x) + get_sz(n->l) + 1;
}
void pull(node *n) {
n->sz = 1 + get_sz(n->l) + get_sz(n->r);
n->sum = n->val + get_sum(n->l) + get_sum(n->r);
n->min_suf = min(get_min_suf(n->r), get_sum(n->r) + n->val + get_min_suf(n->l));
n->max_suf = max(get_max_suf(n->r), get_sum(n->r) + n->val + get_max_suf(n->l));
}
pair<node *, node *> split_sz(node *n, int k) {
if (n == NULL)
return {n, n};
if (k <= get_sz(n->l)) {
auto pp = split_sz(n->l, k);
n->l = pp.second;
pull(n);
return {pp.first, n};
}
auto pp = split_sz(n->r, k - 1 - get_sz(n->l));
n->r = pp.first;
pull(n);
return {n, pp.second};
}
node *merge(node *a, node *b) {
if (a == NULL) {
return b;
}
if (b == NULL)
return a;
if (a->y > b->y) {
a->r = merge(a->r, b);
pull(a);
return a;
}
b->l = merge(a, b->l);
pull(b);
return b;
}
struct mdata {
node *root = NULL;
int size();
int get(int);
};
int mdata::size() { return get_sz(root); }
int mdata::get(int i) { return get_ct(root, i); }
void erase(mdata &mda, int i, int x) {
int ind = getInd(mda.root, i);
auto pp1 = split_sz(mda.root, ind);
auto pp2 = split_sz(pp1.second, 1);
mda.root = merge(pp1.first, pp2.second);
}
void add(mdata &mda, int i, int x) {
int ind = getInd(mda.root, i);
auto pp = split_sz(mda.root, ind);
mda.root = merge(merge(pp.first, new node(i, x)), pp.second);
}
int get_fst_diff(node *n, int mx_diff, int mx, int mn) {
if (n == NULL) {
return 0;
}
int mx1 = max(max(mx, get_sum(n->r) + n->val), get_max_suf(n->r));
int mn1 = min(min(mn, get_sum(n->r) + n->val), get_min_suf(n->r));
if (mx1 - mn1 > mx_diff) {
return get_fst_diff(n->r, mx_diff, mx, mn) + 1 + get_sz(n->l);
}
return get_fst_diff(n, mx_diff, mx1, mn1);
}
int get_fst_diff(mdata &mda, int mx_diff) {
// returns index to first arrow after which everything contains in a segment
// of size mx
int cur = 0;
int mn = 0, mx = 0;
for (int i = mda.size() - 1; i >= 0; i--) {
int a = mda.get(i);
cur += a;
mn = min(mn, cur);
mx = max(mx, cur);
if (mx - mn >= mx_diff) {
return i + 1;
}
}
return 0;
}
int get_min_suf(mdata &mda, int fst) {
auto pp = split_sz(mda.root, fst);
int ans = get_min_suf(pp.second);
mda.root = merge(pp.first, pp.second);
return ans;
int cur = 0;
int mn = 0;
for (int i = mda.size() - 1; i >= fst; i--) {
cur += mda.get(i);
if (cur < mn) {
mn = cur;
}
}
return mn;
}
int get_max_suf(mdata &mda, int fst) {
auto pp = split_sz(mda.root, fst);
int ans = get_max_suf(pp.second);
mda.root = merge(pp.first, pp.second);
return ans;
int cur = 0;
int mn = 0;
for (int i = mda.size() - 1; i >= fst; i--) {
cur += mda.get(i);
if (cur > mn) {
mn = cur;
}
}
return mn;
}
int get_ans(mdata &mda, int mx) {
int fst = get_fst_diff(mda, mx);
int start = 0;
if (fst == 0) {
start = 0;
} else {
int a = mda.get(fst - 1);
if (a > 0) {
start = mx;
} else {
start = 0;
}
}
if (start == 0) {
int ans = get_max_suf(mda, fst);
return start + ans;
} else {
int ans = get_min_suf(mda, fst);
return start + ans;
}
}
vector<int> distribute_candies(vector<int> c, vector<int> l, vector<int> r,
vector<int> v) {
int n = c.size();
vector<vector<int>> add_evs(n + 1), rem_evs(n + 1);
for (int i = 0; i < (int)l.size(); i++) {
add_evs[l[i]].push_back(i);
rem_evs[r[i] + 1].push_back(i);
}
mdata mda;
vector<int> ans;
ans.reserve(n);
for (int i = 0; i < n; i++) {
for (int el : rem_evs[i]) {
erase(mda, el, v[el]);
}
for (int el : add_evs[i]) {
add(mda, el, v[el]);
}
ans.push_back(get_ans(mda, c[i]));
}
return ans;
}
# | 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... |