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 <bits/stdc++.h>
#include "overtaking.h"
#define int ll
using namespace std;
typedef long long ll;
typedef pair <ll, ll> pii;
const ll NMAX = 38000001;
const ll INF = 1e18;
const ll nrbits = 20;
const ll MOD = 998244353;
const pii NEUTRU = {0, 0};
struct Node {
/// max(x + a, b)
int st, dr;
ll maxi; /// basically asta e valoarea lui f(dr)
pii lazy;
ll eval(ll x) {
return max(x + lazy.first, lazy.second);
}
};
Node aint[NMAX];
int cnt;
pii combine(pii nou, pii a) {
return {a.first + nou.first, max(a.second + nou.first, nou.second)};
}
void propaga(int node, ll st, ll dr) {
if(node == -1 || aint[node].lazy == NEUTRU) return;
if(st != dr) {
if(aint[node].st == -1) {
aint[node].st = cnt;
aint[cnt++] = {-1, -1, (st + dr) / 2, NEUTRU};
}
if(aint[node].dr == -1) {
aint[node].dr = cnt;
aint[cnt++] = {-1, -1, dr, NEUTRU};
}
aint[aint[node].st].lazy = combine(aint[node].lazy, aint[aint[node].st].lazy);
aint[aint[node].dr].lazy = combine(aint[node].lazy, aint[aint[node].dr].lazy);
}
aint[node].maxi = aint[node].eval(aint[node].maxi);
aint[node].lazy = NEUTRU;
}
void update(int node, ll st, ll dr, ll a, ll b, pii functie) {
propaga(node, st, dr);
if(a <= st && dr <= b) {
aint[node].lazy = combine(functie, aint[node].lazy);
return;
}
ll mid = (st + dr) / 2;
if(a <= mid) {
if(aint[node].st == -1) {
aint[node].st = cnt;
aint[cnt++] = {-1, -1, mid, NEUTRU};
}
update(aint[node].st, st, mid, a, b, functie);
}
if(b > mid) {
if(aint[node].dr == -1) {
aint[node].dr = cnt;
aint[cnt++] = {-1, -1, dr, NEUTRU};
}
update(aint[node].dr, mid + 1, dr, a, b, functie);
}
propaga(aint[node].st, st, mid);
propaga(aint[node].dr, mid + 1, dr);
if(aint[node].dr != -1)
aint[node].maxi = aint[aint[node].dr].maxi;
else
aint[node].maxi = dr;
/// Calculez valoarea lui f(dr)
}
ll cb(int node, ll st, ll dr, ll val, pii acum) {
acum = combine(acum, aint[node].lazy);
if(st == dr) {
return st;
}
ll mid = (st + dr) / 2;
ll maxist = max(acum.first + mid, acum.second);
if(aint[node].st != -1) {
pii atunci = combine(acum, aint[aint[node].st].lazy);
maxist = max(atunci.first + aint[aint[node].st].maxi, atunci.second);
}
// debugs(st);
// debugs(dr);
// debug(maxist);
if(maxist > val) { /// vrem pe alea care au ajuns strict inaintea mea (ca autobuz cartof) - la cazul de egalitate...ori tu (ca de ala de rezerva) esti mai rapid deci e bine ca nu de updatez...ori esti mai incet...deci nici macar nu te voi afecta daca te updatez
/// mergea si asa, dar pt convenineta am scos pana la urma pe alea mai rapide ca autobuzele de rezerva fiindca nu ne interseaza deloc
if(aint[node].st == -1) {
/// max(acum.first + x, acum.second) == val + 1
int stanga = max(acum.second, acum.first + st);
int dreapta = max(acum.first + dr, acum.second);
val++;
if(val <= stanga) return st;
if(val >= dreapta) return dr;
return val - acum.first;
}
return cb(aint[node].st, st, mid, val, acum);
}
if(aint[node].dr == -1) {
int stanga = max(acum.second, acum.first + st);
int dreapta = max(acum.first + dr, acum.second);
val++;
if(val <= stanga) return st;
if(val >= dreapta) return dr;
return val - acum.first;
}
return cb(aint[node].dr, mid + 1, dr, val, acum);
}
ll query(int node, ll st, ll dr, ll nr, pii acum) {
acum = combine(acum, aint[node].lazy);
if(st == dr) {
return max(acum.first + aint[node].maxi, acum.second);
}
ll mid = (st + dr) / 2;
if(nr <= mid) {
if(aint[node].st == -1)
return max(acum.first + nr, acum.second);
return query(aint[node].st, st, mid, nr, acum);
}
if(aint[node].dr == -1)
return max(acum.first + nr, acum.second);
return query(aint[node].dr, mid + 1, dr, nr, acum);
}
ll panaAcum[1001];
ll total;
void init(signed L, signed N, std::vector<long long> T, std::vector<signed> W, signed X, signed M, std::vector<signed> S) {
vector <int> indici;
total = 1LL * L * X;
aint[cnt++] = {-1, -1, INF, NEUTRU};
for(int i = 0; i < N; i++) {
indici.push_back(i);
panaAcum[i] = T[i];
}
sort(indici.begin(), indici.end(), [&W](const int &a, const int &b) {
return W[a] > W[b];
});
while(indici.size() && W[indici.back()] <= X) {
indici.pop_back();
}
for(int i = 1; i < M; i++) { /// de la 1
//debug(i);
sort(indici.begin(), indici.end(), [&W, &panaAcum](const int &a, const int &b) {
return ((pii){panaAcum[a], W[a]} < (pii){panaAcum[b], W[b]});
});
ll maxim = 0;
vector <ll> unde; /// pt fiecare indice calculam pe cine afecteaza
for(int j = 0; j < indici.size(); j++) {
unde.push_back(cb(0, 0, INF, panaAcum[indici[j]], NEUTRU));
}
for(int j = 0; j < indici.size(); j++) {
panaAcum[indici[j]] += 1LL * W[indici[j]] * (S[i] - S[i - 1]);
maxim = max(maxim, panaAcum[indici[j]]);
panaAcum[indici[j]] = maxim;
panaAcum[indici[j]] -= X * (S[i] - S[i - 1]);
// debugs(indici[j]);
// debug(panaAcum[indici[j]]);
// debug(unde[j]);
update(0, 0, INF, unde[j], INF, {0, panaAcum[indici[j]]});
}
}
}
long long arrival_time(long long Y) {
return query(0, 0, INF, Y, NEUTRU) + total;
}
Compilation message (stderr)
overtaking.cpp: In function 'void init(int, int, std::vector<long long int>, std::vector<int>, int, int, std::vector<int>)':
overtaking.cpp:154:50: warning: capture of variable 'panaAcum' with non-automatic storage duration
154 | sort(indici.begin(), indici.end(), [&W, &panaAcum](const int &a, const int &b) {
| ^~~~~~~~
overtaking.cpp:135:4: note: 'll panaAcum [1001]' declared here
135 | ll panaAcum[1001];
| ^~~~~~~~
overtaking.cpp:159:26: warning: comparison of integer expressions of different signedness: 'll' {aka 'long long int'} and 'std::vector<long long int>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
159 | for(int j = 0; j < indici.size(); j++) {
| ~~^~~~~~~~~~~~~~~
overtaking.cpp:163:26: warning: comparison of integer expressions of different signedness: 'll' {aka 'long long int'} and 'std::vector<long long int>::size_type' {aka 'long unsigned int'} [-Wsign-compare]
163 | for(int j = 0; j < indici.size(); j++) {
| ~~^~~~~~~~~~~~~~~
# | 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... |