[HNOI2011]括号修复

\(nd[4]\)

0——多出来的右括号
1——多出来的左括号
2——取反后多出来的右括号
3——取反后多出来的左括号

这样一来

Swap: swap(0,3),swap(1,2),swap(sn[0],sn[1])
Invert: swap(0,2),swap(1,3),val[k]^=1
Replace: v<-siz[k],v+2^1<-siz[k],v^1<-0,v+2<-0

注意一下运算优先级就好

#include"cstdio"
#include"cstring"
#include"iostream"
#include"algorithm"
using namespace std;

const int MAXN=1e5+5;

int n,m,cnt,root;
char ch[MAXN];
int val[MAXN],rev[MAXN],sn[2][MAXN],siz[MAXN];
int nd[4][MAXN],tag3[MAXN];
bool tag1[MAXN],tag2[MAXN];

int cret(int v)
{
	int tmp=++cnt;
	siz[tmp]=1;
	val[tmp]=v;
	tag3[tmp]=-1;
	nd[v][tmp]=nd[v+2^1][tmp]=1;
	rev[tmp]=rand();
	return tmp;
}

void pushdown(int k)
{
	if(tag3[k]!=-1){
		if(sn[0][k]) nd[tag3[k]][sn[0][k]]=nd[tag3[k]+2^1][sn[0][k]]=siz[sn[0][k]],nd[tag3[k]^1][sn[0][k]]=nd[tag3[k]+2][sn[0][k]]=0,val[sn[0][k]]=tag3[sn[0][k]]=tag3[k],tag1[sn[0][k]]=tag2[sn[0][k]]=0;
		if(sn[1][k]) nd[tag3[k]][sn[1][k]]=nd[tag3[k]+2^1][sn[1][k]]=siz[sn[1][k]],nd[tag3[k]^1][sn[1][k]]=nd[tag3[k]+2][sn[1][k]]=0,val[sn[1][k]]=tag3[sn[1][k]]=tag3[k],tag1[sn[1][k]]=tag2[sn[1][k]]=0;
	}if(tag2[k]){
		swap(sn[0][k],sn[1][k]);
		if(sn[0][k]) swap(nd[0][sn[0][k]],nd[3][sn[0][k]]),swap(nd[1][sn[0][k]],nd[2][sn[0][k]]),tag2[sn[0][k]]^=1;
		if(sn[1][k]) swap(nd[0][sn[1][k]],nd[3][sn[1][k]]),swap(nd[1][sn[1][k]],nd[2][sn[1][k]]),tag2[sn[1][k]]^=1;
	}if(tag1[k]){
		if(sn[0][k]) swap(nd[0][sn[0][k]],nd[2][sn[0][k]]),swap(nd[1][sn[0][k]],nd[3][sn[0][k]]),tag1[sn[0][k]]^=1,val[sn[0][k]]^=1;
		if(sn[1][k]) swap(nd[0][sn[1][k]],nd[2][sn[1][k]]),swap(nd[1][sn[1][k]],nd[3][sn[1][k]]),tag1[sn[1][k]]^=1,val[sn[1][k]]^=1;
	}tag1[k]=tag2[k]=0;
	tag3[k]=-1;
	return;
}

void pushup(int k)
{
	siz[k]=siz[sn[0][k]]+siz[sn[1][k]]+1;
	nd[0][k]=nd[0][sn[0][k]];
	nd[1][k]=nd[1][sn[1][k]];
	int tmp=nd[1][sn[0][k]]-nd[0][sn[1][k]];
	tmp+=val[k]?1:-1;
	if(tmp>0) nd[1][k]+=tmp;
	else nd[0][k]-=tmp;
	
	nd[2][k]=nd[2][sn[0][k]];
	nd[3][k]=nd[3][sn[1][k]];
	tmp=nd[3][sn[0][k]]-nd[2][sn[1][k]];
	tmp+=val[k]?-1:1;
	if(tmp>0) nd[3][k]+=tmp;
	else nd[2][k]-=tmp;
	return;
}

void dro(int k,int v,int &x,int &y)
{
	if(!k){x=y=0;return;}
	pushdown(k);
	if(siz[sn[0][k]]<v) x=k,dro(sn[1][k],v-siz[sn[0][k]]-1,sn[1][k],y);
	else y=k,dro(sn[0][k],v,x,sn[0][k]);
	pushup(k);
	return;
}

int un(int x,int y)
{
	if(!x||!y) return x|y;
	if(rev[x]<rev[y]){
		pushdown(x);
		sn[1][x]=un(sn[1][x],y);
		pushup(x);
		return x;
	}pushdown(y);
	sn[0][y]=un(x,sn[0][y]);
	pushup(y);
	return y;
}

void slv0(int l,int r)
{
	int x,y,z;
	dro(root,r,x,z);
	dro(x,l-1,x,y);
	printf("%d\n",(nd[0][y]+1)/2+(nd[1][y]+1)/2);
	root=un(un(x,y),z);
	return;
}

void slv1(int l,int r)
{
	int x,y,z;
	dro(root,r,x,z);
	dro(x,l-1,x,y);
	tag1[y]^=1;val[y]^=1;
	pushdown(y);pushup(y);
	root=un(un(x,y),z);
	return;
}

void slv2(int l,int r)
{
	int x,y,z;
	dro(root,r,x,z);
	dro(x,l-1,x,y);
	tag2[y]^=1;
	pushdown(y);pushup(y);
	root=un(un(x,y),z);
	return;
}

void slv3(int l,int r,int kd)
{
	int x,y,z;
	dro(root,r,x,z);
	dro(x,l-1,x,y);
	tag3[y]=val[y]=kd;tag1[y]=tag2[y]=0;
	pushdown(y);pushup(y);
	root=un(un(x,y),z);
	return;
}

int main()
{
	scanf("%d%d",&n,&m);
	scanf("%s",ch+1);
	for(int i=1;i<=n;++i) root=un(root,cret(ch[i]=='('));
	while(m--){
		int l,r;
		scanf("%s%d%d",ch+1,&l,&r);
		if(ch[1]=='Q') slv0(l,r);
		else if(ch[1]=='I') slv1(l,r);
		else if(ch[1]=='S') slv2(l,r);
		else scanf("%s",ch+1),slv3(l,r,ch[1]=='(');
	}return 0;
}
posted @ 2019-02-11 21:44  A·H  阅读(202)  评论(0编辑  收藏  举报