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>
#define X first
#define Y second
#define all(x) begin(x), end(x)
#define FOR(i, a, b) for(int i = (a); i <= (b); i++)
#define FORD(i, b, a) for(int i = (b); i >= (a); i--)
#define REP(i, a, b) for (int i = (a); i < (b); i++)
#define mxx max_element
#define mnn min_element
#define SQR(x) (1LL * (x) * (x))
#define MASK(i) (1LL << (i))
#define Point Vector
#define left Left
#define right Right
#define div Div
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
typedef double db;
typedef long double ld;
typedef pair<db, db> pdb;
typedef pair<ld, ld> pld;
typedef pair<int, int> pii;
typedef pair<int, pii> piii;
typedef pair<ll, ll> pll;
typedef pair<ll, pll> plll;
typedef pair<ll, int> pli;
typedef pair<ll, pii> plii;
template<class A, class B>
bool maximize(A& x, B y) {
if (x < y) return x = y, true; else return false;
}
template<class A, class B>
bool minimize(A& x, B y) {
if (x > y) return x = y, true; else return false;
}
/* END OF TEMPLATE */
const int N = 150007;
const int M = 407;
int n, L, a[N], Q, x[N], pos[N];
int S, numBlock = 0;
struct BLOCK {
pii b[2 * M];
pii f[2 * M];
int sz;
BLOCK() {
REP(i, 0, 2 * M) b[i] = {0, 0};
sz = 0;
}
} block[M];
void print_block() {
REP(i, 0, numBlock) {
cout<<"block "<<i<<":\n";
REP(j, 0, block[i].sz) cout<<block[i].b[j].X<<" "<<block[i].b[j].Y<<"\n";
cout<<"array F : \n";
REP(j, 0, block[i].sz) cout<<block[i].f[j].X<<" "<<block[i].f[j].Y<<"\n";
cout<<"\n";
}
}
void print_block(BLOCK& a) {
cout<<"array b : \n";
REP(j, 0, a.sz) cout<<a.b[j].X<<" "<<a.b[j].Y<<"\n";
cout<<"array F : \n";
REP(j, 0, a.sz) cout<<a.f[j].X<<" "<<a.f[j].Y<<"\n";
cout<<"\n";
}
void init_block_data(BLOCK& a) {
int j = a.sz - 1;
FORD(i, a.sz - 1, 0) {
while (j > i && a.b[i].X + L < a.b[j].X) j--;
if (j == a.sz - 1) a.f[i] = {1, a.b[i].X + L}; else a.f[i] = {a.f[j + 1].X + 1, a.f[j + 1].Y};
}
}
void init(int nn, int LL, int x[]) {
n = nn;
L = LL;
REP(i, 0, n) a[i] = x[i];
S = sqrt(n);
for (int i = 0; i < n; i+=S) {
FOR(j, i, min(n - 1, i + S - 1)) {
block[numBlock].b[block[numBlock].sz++] = {a[j], j};
pos[j] = numBlock;
}
init_block_data(block[numBlock]);
numBlock++;
}
// print_block();
}
pii c[N];
void rebuild_block() {
int m = 0;
REP(i, 0, numBlock)
REP(j, 0, block[i].sz) c[m++] = block[i].b[j];
numBlock = 0;
for (int i = 0; i < n; i+=S) {
block[numBlock].sz = 0;
FOR(j, i, min(n - 1, i + S - 1)) {
block[numBlock].b[block[numBlock].sz++] = c[j];
pos[c[j].Y] = numBlock;
}
init_block_data(block[numBlock]);
numBlock++;
}
}
void delete_element_in_block(BLOCK& a, int pos) {
// x la vi tri
REP(i, 0, a.sz)
if (a.b[i].Y == pos) {
// print_block(a);
REP(j, i, a.sz - 1) a.b[j] = a.b[j + 1];
a.sz--;
// print_block(a);
break;
}
init_block_data(a);
}
void add_element(BLOCK& a, int pos, int val) {
if (a.sz == 0) {
a.b[a.sz++] = {val, pos};
}
else
if (val <= a.b[0].X) {
FORD(j, a.sz, 1) a.b[j] = a.b[j - 1];
a.b[0] = {val, pos};
a.sz++;
}
else
if (val >= a.b[a.sz - 1].X) {
a.b[a.sz] = {val, pos};
a.sz++;
}
else {
REP(i, 1, a.sz)
if (val <= a.b[i].X) {
FORD(j, a.sz, i + 1) a.b[j] = a.b[j - 1];
a.b[i] = {val, pos};
a.sz++;
break;
}
}
// print_block(a);
init_block_data(a);
}
void add_element(int POS, int val) {
REP(i, 0, numBlock)
if (val <= block[i].b[block[i].sz - 1].X || i == numBlock - 1) {
pos[POS] = i;
add_element(block[i], POS, val);
break;
}
REP(i, 0, numBlock)
if (block[i].sz > 2 * S) {
rebuild_block();
break;
}
}
void update_block(int i, int y) {
delete_element_in_block(block[pos[i]], i);
// exit(0);
// exit(0);
a[i] = y;
add_element(i, a[i]);
// print_block();
}
int get_answer() {
int pos_not_empty = 0;
REP(i, 0, numBlock)
if (block[i].sz > 0) {
pos_not_empty = i;
break;
}
int res = block[pos_not_empty].f[0].X, tmp = block[pos_not_empty].f[0].Y;
REP(i, pos_not_empty + 1, numBlock) {
if (tmp >= block[i].b[block[i].sz - 1].X) continue;
int d = 0, c = (int)block[i].sz - 1, g, ans = -1;
while (d <= c) {
g = (d + c) >> 1;
if (tmp >= block[i].b[g].X) ans = g, d = g + 1; else c = g - 1;
}
ans++;
res+=block[i].f[ans].X;
tmp = block[i].f[ans].Y;
}
return res;
}
int update(int i, int y) {
update_block(i, y);
return get_answer();
}
# | 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... |