给定N个点,点的编号是从1 ~ N,M次‘Q’或‘D或‘R’的操作,“D x”表示的是破坏这个点x,“R”是修复之前破坏的点,“Q x”表示询问点x所在连续区间的长度。假定最初每个点都是好的。
首先对操作进行分析:可以把D操作看成把区间在点x出截断,R操作是把x左右【包括点x】进行合并。每次合并两个区间 [a,b] , [c,d] 的时候,得到的新的区间[a,d] 的最大连续区间长度为:
Len(a,d) = max{ Len(a,b),Len(c,d) ,End(b) + Begin(c) }; <==有一点分治的思想
End(b)表示以b结尾的最大连续区间长度,Begin(c)表示以c开头的最大连续区间长度,这里我们可以知道,合并两个区间,并不是简单的区间加减,而是还要保持一个以区间第一个元素开始的最大连续区间长度以及 以区间最后一个元素结尾的最大连续区间长度。
那么,现在问题就比较好办了,线段树每个节点包含三个信息,分别是以区间第一个元素开始的最大连续区间长度ln,以区间最后一个元素结尾的最大连续区间长度rn,以及区间最大连续长度mn。
接下来,便是Q操作了,询问点x 所在连续区间的长度。在递归左儿子节点的时候,注意一下,x是否在以左儿子区间最后一个元素结尾长度为左儿子rn的范围里面,如果在,那么x所在的连续区间便有可能包含右儿子的一部分,递归右儿子的时候,同理!
/****************************>>>>HEADFILES<<<<****************************/
#include
#include
一年之后,再写了一遍这个题目。感觉代码风格还是变化了一点。再贴个代码~
#include
using namespace std;
const int MX = 5e4 + 5;
#define lch (rt << 1)
#define rch (rt << 1 | 1)
typedef pair PII;
struct Seg {
/**
* status:
* 0 -- Bad
* 1 -- Good
* -1 -- not cover
* I: 节点对应区间两端点以及两短点的标号
*/
int sum, status;
PII I[2];
} seg[MX * 3];
int N, M, x;
char oper[5];
inline void pushUp(int rt) {
if(seg[lch].status == seg[rch].status) seg[rt].status = seg[lch].status;
else seg[rt].status = -1;
if(seg[lch].status == 1 && seg[rch].status == 1) seg[rt].sum = seg[lch].sum + seg[rch].sum;
else seg[rt].sum = 0;
}
void build(int l, int r, int rt) {
int temp;
if(l == r) {
seg[rt].sum = seg[rt].status = 1;
seg[rt].I[0] = make_pair(l, rt);
seg[rt].I[1] = make_pair(r, rt);
return;
}
int mid = (l + r) >> 1;
build(l, mid, lch);
build(mid + 1, r, rch);
pushUp(rt);
seg[rt].I[0] = seg[lch].I[0];
seg[rt].I[1] = seg[rch].I[1];
}
void update(int pos, int v, int l, int r, int rt) {
if(l == r) {
seg[rt].status = v;
seg[rt].sum = (v == 1) ? 1 : 0;
return;
}
int mid = (l + r) >> 1;
if(pos <= mid) update(pos, v, l, mid, lch);
else update(pos, v, mid + 1, r, rch);
pushUp(rt);
}
int query_status(int pos, int l, int r, int rt) {
if(l == r) return seg[rt].status;
int mid = (l + r) >> 1;
if(pos <= mid) return query_status(pos, l, mid, lch);
else return query_status(pos, mid + 1, r, rch);
}
int query(int pos, int l, int r, int rt, int d) {
if(seg[rt].status == 0) return 0;
int mid = (l + r) >> 1, ret = 0;
if(seg[rt].status == 1) {
ret += seg[rt].sum;
PII &lb = seg[rt].I[0], &ub = seg[rt].I[1];
if(lb.first != 1 && seg[lb.second].status == 1 && d != 1)
ret += query(lb.first - 1, 1, N, 1, -1);
if(ub.first != N && seg[ub.second].status == 1 && d != -1)
ret += query(ub.first + 1, 1, N, 1, 1);
return ret;
}
if(pos <= mid) {
ret += query(pos, l, mid, lch, d);
} else {
ret += query(pos, mid + 1, r, rch, d);
}
return ret;
}
int main() {
// freopen("input.txt", "r", stdin);
while(~scanf("%d %d", &N, &M)) {
build(1, N, 1);
stack buf;
while(M --) {
scanf("%s", oper);
if(oper[0] == 'D') {
scanf("%d", &x); buf.push(x);
update(x, 0, 1, N, 1);
} else if(oper[0] == 'Q') {
scanf("%d", &x);
int k = query_status(x, 1, N, 1), ans = 0;
if(k != 0) {
ans = query(x, 1, N, 1, 0);
}
printf("%d\n", ans);
} else {
if(buf.empty()) continue;
x = buf.top(); buf.pop();
update(x, 1, 1, N, 1);
}
}
}
return 0;
}