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 "holiday.h"
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef pair<int, int> pii;
struct SegmentTree {
int n, h;
vector<ll> base;
vector<int> cnt;
vector<ll> sum;
SegmentTree() {}
SegmentTree(int _n, const vector<ll>& arr) : n(_n) {
h = Log2(n);
n = 1 << h;
base.resize(2*n, 0);
for (int i=0; i<arr.size(); ++i) base[n+i] = arr[i];
cnt.resize(2*n, 0);
sum.resize(2*n, 0);
}
void update(int p, int v) {
p += n;
cnt[p] += v;
sum[p] += base[p] * v;
for (p/=2; p>0; p/=2) {
cnt[p] = cnt[2*p] + cnt[2*p+1];
sum[p] = sum[2*p] + sum[2*p+1];
}
}
ll query(int k) {
if (cnt[1] <= k) return sum[1];
ll ret = 0;
int curr = 1;
while (curr < n) {
if (cnt[2*curr+1] <= k) {
ret += sum[2*curr+1];
k -= cnt[2*curr+1];
curr = 2*curr;
} else {
curr = 2*curr+1;
}
}
return ret;
}
static int Log2(int x){
int ret = 0;
while (x > (1 << ret)) ret++;
return ret;
}
};
static const int MAX_N = 1e5 + 1;
static ll ans = 0;
static int n, start, d;
static int arr[MAX_N];
static SegmentTree seg;
static pii curr_range;
static void move(int l, int r) {
for (int& x = curr_range.first; x < l; ++x) seg.update(arr[x], -1);
for (int& x = curr_range.first; x > l; --x) seg.update(arr[x-1], 1);
for (int& x = curr_range.second; x < r; ++x) seg.update(arr[x+1], 1);
for (int& x = curr_range.second; x > r; --x) seg.update(arr[x], -1);
curr_range = { l, r };
}
static void test() {
for (int i=0; i<n; ++i) cout << seg.query(i) << ' ';
cout << endl;
}
static void solve(int ls, int le, int rs, int re) {
if (ls > le) return;
assert(rs <= re);
pii past_range = curr_range;
int m = (ls + le) / 2;
ll opt_val = -1;
int opt_idx = -1;
move(m, rs);
// cout << "visit: " << ls << ' ' << le << ' ' << rs << ' ' << re << endl;
for (int i=rs; i<=re; ++i) {
// test();
ll tmp = seg.query(d-2*(start-m)-(i-start));
if (opt_val < tmp) {
opt_val = tmp;
opt_idx = i;
}
if (i < re) seg.update(arr[i+1], 1);
}
ans = max(ans, opt_val);
// cout << m << ' ' << opt_val << ' ' << opt_idx << endl;
curr_range = { m, re };
solve(ls, m-1, rs, opt_idx);
solve(m+1, le, opt_idx, re);
move(past_range.first, past_range.second);
}
ll findMaxAttraction(int N, int Start, int D, int attraction[]) {
n = N;
start = Start;
d = D;
vector<ll> attract(n);
for (int i=0; i<n; ++i) attract[i] = attraction[i];
for (int i=0; i<n; ++i) arr[i] = attract[i];
sort(attract.begin(), attract.end());
seg = SegmentTree(n, attract);
for (int i=0; i<n; ++i) arr[i] = lower_bound(attract.begin(), attract.end(), arr[i]) - attract.begin();
seg.update(arr[start], 1);
curr_range = { start, start };
solve(0, start, start, n-1);
reverse(arr, arr+n);
start = n-1-start;
curr_range = { start, start };
solve(0, start, start, n-1);
return ans;
}
Compilation message (stderr)
holiday.cpp: In constructor 'SegmentTree::SegmentTree(int, const std::vector<long long int>&)':
holiday.cpp:17:24: warning: comparison of integer expressions of different signedness: 'int' and 'std::vector<long long int>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
17 | for (int i=0; i<arr.size(); ++i) base[n+i] = arr[i];
| ~^~~~~~~~~~~
holiday.cpp: At global scope:
holiday.cpp:72:13: warning: 'void test()' defined but not used [-Wunused-function]
72 | static void test() {
| ^~~~
# | 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... |