# | Time | Username | Problem | Language | Result | Execution time | Memory |
---|---|---|---|---|---|---|---|
36424 | cheater2k | UFO (IZhO14_ufo) | C++14 | 1566 ms | 262144 KiB |
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>
using namespace std;
const int N = 1e6 + 5;
struct SegmentTree {
int *T;
int sz;
void init(int _sz=0) {
sz = _sz;
T = new int[4 * (sz + 10)];
}
#define mid ((l + r) >> 1)
void upd(int v, int l, int r, int pos, int val) {
if (l > r || l > pos || r < pos) return;
if (l == r) { T[v] = val; return; }
upd(v << 1, l, mid, pos, val); upd(v << 1 | 1, mid + 1, r, pos, val);
T[v] = max(T[v << 1], T[v << 1 | 1]);
}
int getL(int v, int l, int r, int L, int val) { // [L,sz]
// find the smallest i such as i >= L and a[i] >= val
if (r < L || l > r) return sz + 1;
if (T[v] < val) return sz + 1;
if (l == r) {
if (T[v] >= val) return l; else return sz + 1;
}
int lef = getL(v << 1, l, mid, L, val);
if (lef < sz + 1) return lef;
else return getL(v << 1 | 1, mid + 1, r, L, val);
}
int getR(int v, int l, int r, int R, int val) { // [1,R]
// find the greatest i such as i <= R and a[i] >= val
if (l > R || l > r) return 0;
if (T[v] < val) return 0;
if (l == r) {
if (T[v] >= val) return l; else return 0;
}
int rig = getR(v << 1 | 1, mid + 1, r, R, val);
if (rig > 0) return rig;
else return getR(v << 1, l, mid, R, val);
}
#undef mid
void upd(int pos, int val) { upd(1, 1, sz, pos, val); }
int getL(int pos, int val) { return getL(1, 1, sz, pos, val); }
int getR(int pos, int val) { return getR(1, 1, sz, pos, val); }
} row[N], col[N];
int n, m, R, k, p;
int *a[N];
long long *sum[N];
long long ans;
void change(int r, int c) {
a[r][c]--;
row[r].upd(c, a[r][c]);
col[c].upd(r, a[r][c]);
}
int main() {
ios_base::sync_with_stdio(false); cin.tie(0);
cin >> n >> m >> R >> k >> p;
for (int i = 1; i <= n; ++i) {
a[i] = new int[m + 1];
for (int j = 1; j <= m; ++j) cin >> a[i][j];
}
for (int i = 1; i <= n; ++i) {
row[i].init(m);
}
for (int i = 1; i <= m; ++i) {
col[i].init(n);
}
for (int i = 1; i <= n; ++i) {
for (int j = 1; j <= m; ++j) {
row[i].upd(j, a[i][j]);
col[j].upd(i, a[i][j]);
}
}
// process queries
while(k--) {
char dir; int x, height;
cin >> dir >> x >> height;
if (dir == 'W') {
// row, getL
int pos = 1, cnt = R;
while(cnt > 0 && pos <= m) {
pos = row[x].getL(pos, height); // [pos...m], >= height
if (pos == m + 1) break;
change(x, pos);
--cnt; ++pos;
}
} else if (dir == 'E') {
// row, getR
int pos = m, cnt = R;
while(cnt > 0 && pos >= 1) {
pos = row[x].getR(pos, height); // [1...pos], >= height
if (pos == 0) break;
change(x, pos);
--cnt; --pos;
}
} else if (dir == 'N') {
// col, getL
int pos = 1, cnt = R;
while(cnt > 0 && pos <= n) {
pos = col[x].getL(pos, height); // [pos...n], >= height
if (pos == n + 1) break;
change(pos, x);
--cnt; ++pos;
}
} else { // dir == 'S'
// col, getR
int pos = m, cnt = R;
while(cnt > 0 && pos >= 1) {
pos = col[x].getR(pos, height); // [1...pos], >= height
if (pos == 0) break;
change(pos, x);
--cnt; --pos;
}
}
}
// calculate sum
sum[0] = new long long[m + 1];
for (int i = 1; i <= n; ++i) {
sum[i] = new long long[m + 1];
for (int j = 1; j <= m; ++j) {
sum[i][j] = sum[i-1][j] + sum[i][j-1] - sum[i-1][j-1] + a[i][j];
}
}
for (int i = p; i <= n; ++i) {
for (int j = p; j <= m; ++j) ans = max(ans, sum[i][j] - sum[i-p][j] - sum[i][j-p] + sum[i-p][j-p]);
}
printf("%lld\n", ans);
}
# | Verdict | Execution time | Memory | Grader output |
---|---|---|---|---|
Fetching results... |