loj #6515. 「雅礼集训 2018 Day10」贪玩蓝月

\(\color{#0066ff}{输入样例}\)

0
11 10
QU 0 0
QU 1 9
IG 14 7
IF 3 5
QU 0 9
IG 1 8
DF
QU 0 4
IF 1 2
DG
QU 2 9

\(\color{#0066ff}{输出样例}\)

0
-1
12
8
9

\(\color{#0066ff}{数据范围与提示}\)

\(\color{#0066ff}{题解}\)

维护两个栈,一个是前面插入,一个是后面插入,每次插入的时候跑一遍背包

删除的时候,如果一个栈空了,那么把另一个栈的一半弄过来(保证复杂度),暴力处理一下DP

对于询问,对于每一个\(x\in[1,mod]\),我们可以找到一个区间使得\([(L+x)\%mod,(R+x)\%mod]就是给的询问的[l,r]\)

我们只需要一个栈的DP数组的x位置的值和另一个数组一段区间的最大值,用这个更新答案就行

#include<bits/stdc++.h>
#define LL long long
LL in() {
	char ch; LL x = 0, f = 1;
	while(!isdigit(ch = getchar()))(ch == '-') && (f = -f);
	for(x = ch ^ 48; isdigit(ch = getchar()); x = (x << 1) + (x << 3) + (ch ^ 48));
	return x * f;
}
template<typename T> T chkmax(T &a, const T &b) { return a < b? a = b : b; }
template<typename T> T chkmin(T &a, const T &b) { return a < b? a : a = b; }
const int maxn = 5e4 + 10;
const int maxm = 505;
LL st[maxn][20], f[2][maxn][maxm];
int top[2], mod, lg[maxn], inf;
std::pair<LL, LL> t[2][maxn];
char opt[22];
void predoit() {
	memset(f, 0xcf, sizeof f);
	inf = -f[0][0][0];
	f[0][0][0] = f[1][0][0] = 0;
	lg[0] = -1;
	for(int i = 1; i < maxn; i++) lg[i] = lg[i >> 1] + 1;
}
void ins(bool k, int now, std::pair<LL, LL> mp) {
	t[k][now] = mp;
	for(int i = 0; i < mod; i++) f[k][now][i] = f[k][now - 1][i];
	for(int i = 0; i < mod; i++) chkmax(f[k][now][(i + mp.first) % mod], f[k][now - 1][i] + mp.second);
}
void del(bool k) {
	if(top[k]) return (void)(top[k]--);
	int mid = (top[k ^ 1] + 1) >> 1;
	for(int i = 1; i <= mid; i++) t[k][mid - i + 1] = t[k ^ 1][i], t[k ^ 1][i] = t[k ^ 1][i + mid];
	top[k] = mid - 1, top[k ^ 1] = top[k ^ 1] & 1? mid - 1 : mid;
	for(int i = 1; i <= top[k]; i++) ins(k, i, t[k][i]);
	for(int i = 1; i <= top[k ^ 1]; i++) ins(k ^ 1, i, t[k ^ 1][i]);
}
LL getans(int l, int r) {
	int len = lg[r - l + 1];
	return std::max(st[l][len], st[r - (1 << len) + 1][len]);
}

LL query(int l, int r) {
	LL ans = -inf;
	for(int i = 0; i < mod; i++) st[i][0] = f[0][top[0]][i];
	for(int j = 1; j <= lg[mod]; j++)
		for(int i = 0; i + (1 << j) <= mod; i++)
			st[i][j] = std::max(st[i][j - 1], st[i + (1 << (j - 1))][j - 1]);
	for(int i = 0; i < mod; i++) {
		if(f[1][top[1]][i] < 0) continue;
		int L = l - i, R = r - i;
		(L += mod) %= mod, (R += mod) %= mod;
		if(L <= R) chkmax(ans, f[1][top[1]][i] + getans(L, R));
		else chkmax(ans, f[1][top[1]][i] + std::max(getans(L, mod - 1), getans(0, R)));
	}
	return ans < 0? -1 : ans;
};

int main() {
	in(); int T = in(); LL x, y; mod = in();
	predoit();
	while(T --> 0) {
		scanf("%s", opt);
		if(opt[0] == 'I') x = in(), y = in(), ins(opt[1] == 'G', ++top[opt[1] == 'G'], std::make_pair(x % mod, y));
		if(opt[0] == 'D') del(opt[1] == 'G');
		if(opt[0] == 'Q') x = in(), y = in(), printf("%lld\n", query(x, y));
	}
	return 0;
}
posted @ 2019-03-31 07:12  olinr  阅读(654)  评论(0编辑  收藏  举报