【BZOJ-2329&2209】括號修復&括號序列 Splay

2329: [HNOI2011]括號修復

Time Limit: 40 Sec  Memory Limit: 128 MB
Submit: 1007  Solved: 476
[Submit][Status][Discuss]

Description

Input

Output

Sample Input

Sample Output

HINT

Source

Solution

一眼Splay麼....關鍵是怎麼維護...php

一開始看錯題了...想了好久歪路,其實想一下仍是能夠想到的麼..node

對於一個括號序列,首先合法的括號對是對詢問答案沒有貢獻的,因此能夠忽略,剩餘的括號序列必然是形如$)*(*$的序列。ios

因此答案顯然就是$\lfloor \frac{Num_{)}+1}{2} \rfloor + \lfloor \frac{Num{(}+1}{2} \rfloor$。ui

而後只須要Splay中維護左右起最大連續$($左右起最大連續$)$,便可獲得答案。spa

考慮怎麼維護這樣的東西..括號序列一種最典型的表示方式能夠認爲是$+1-1$,這裏也同樣啊,不妨令$(=-1$,$)=+1$blog

這樣就能夠以一種相似最大最小連續子段和的方式去維護上述量了。ip

而後就是修改操做。get

覆蓋操做顯然會清空翻轉操做和取反操做,問題在於取反操做和覆蓋操做的下放順序會對結果有影響!!!(畫圖考慮一下)string

因此值得注意的就是,在下放標記的時候,取反標記一樣會使覆蓋標記發生取反。(再畫圖考慮一下)it

Code

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
using namespace std;
inline int read()
{
	int x=0,f=1; char ch=getchar();
	while (ch<'0' || ch>'9') {if (ch=='-') f=-1; ch=getchar();}
	while (ch>='0' && ch<='9') {x=x*10+ch-'0'; ch=getchar();}
	return x*f;
}
#define MAXN 100010
#define INF 0x3fffffff 
int N,M;
char s[MAXN];

namespace SplayTree{
#define lson(x) son[x][0]
#define rson(x) son[x][1]
	int son[MAXN][2],fa[MAXN],root,sz;
	int lmin[MAXN],rmin[MAXN],lmax[MAXN],rmax[MAXN],size[MAXN],sum[MAXN],val[MAXN];
	int rev[MAXN],inv[MAXN],cov[MAXN];
	inline void Newnode(int &x,int last,int v)
	{
		x=++sz;
		size[x]=1; 
		val[x]=sum[x]=v;
		if (v==1)
			lmin[x]=rmin[x]=0,lmax[x]=rmax[x]=v;
		else
			lmin[x]=rmin[x]=v,lmax[x]=rmax[x]=0;
		fa[x]=last; son[x][0]=son[x][1]=0;
	}
	inline int Right(int x) {return son[fa[x]][1]==x;}
	inline void Update(int x)
	{
		if (!x) return;
		sum[x]=sum[lson(x)]+sum[rson(x)]+val[x];
		size[x]=size[lson(x)]+size[rson(x)]+1;
		lmax[x]=max(lmax[lson(x)],sum[lson(x)]+val[x]+lmax[rson(x)]);
		rmax[x]=max(rmax[rson(x)],sum[rson(x)]+val[x]+rmax[lson(x)]);
		lmin[x]=min(lmin[lson(x)],sum[lson(x)]+val[x]+lmin[rson(x)]);
		rmin[x]=min(rmin[rson(x)],sum[rson(x)]+val[x]+rmin[lson(x)]);
	}
	inline void Rev(int x)
	{
		if (!x) return;
		rev[x]^=1;
		swap(son[x][0],son[x][1]);
		swap(lmax[x],rmax[x]),swap(lmin[x],rmin[x]);
	}
	inline void Cov(int x,int d)
	{
		if (!x) return;
		cov[x]=d; rev[x]=0; inv[x]=0;
		sum[x]=d*size[x]; val[x]=d;
		if (d==1)
			lmax[x]=rmax[x]=size[x],lmin[x]=rmin[x]=0;
		else
			lmax[x]=rmax[x]=0,lmin[x]=rmin[x]=-size[x];
	}
	inline void Inv(int x)
	{
		if (!x) return;
		inv[x]^=1;
		sum[x]=-sum[x]; val[x]=-val[x];
		swap(lmax[x],lmin[x]),swap(rmax[x],rmin[x]);
		lmax[x]=-lmax[x],rmax[x]=-rmax[x];
		lmin[x]=-lmin[x],rmin[x]=-rmin[x];
		cov[x]=-cov[x];
	}
	inline void Pushdown(int x)
	{
		if (rev[x]!=0)
			Rev(son[x][0]),Rev(son[x][1]),rev[x]^=1;
		if (inv[x]!=0)
			Inv(son[x][0]),Inv(son[x][1]),inv[x]^=1;
		if (cov[x]!=0)
			Cov(son[x][0],cov[x]),Cov(son[x][1],cov[x]),cov[x]=0;
	}
	inline int Build(int l,int r,int last)
	{
		int mid=(l+r)>>1,x;
		Newnode(x,last,s[mid]=='('? -1:1);
		if (mid-1>=l) son[x][0]=Build(l,mid-1,x);
		if (mid+1<=r) son[x][1]=Build(mid+1,r,x);
		Update(x);
		return x; 
	}
	inline void Rotate(int x)
	{
		int y=fa[x],z=fa[y],w=Right(x);
		Pushdown(y); Pushdown(x);
		son[y][w]=son[x][w^1]; fa[son[y][w]]=y;
		fa[y]=x; son[x][w^1]=y; fa[x]=z;
		if (z) son[z][son[z][1]==y]=x;
		Update(y); Update(x);
	}
	inline void Splay(int x,int tar)
	{
		for (int y; (y=fa[x])!=tar; Rotate(x))
			if (fa[y]!=tar) Rotate(Right(x)==Right(y) ? y:x);
		if (!tar) root=x;
	}
	inline int Find(int x,int k)
	{
		Pushdown(x);
		if (size[son[x][0]]>=k) return Find(son[x][0],k);
		if (size[son[x][0]]+1==k) return x;
		return Find(son[x][1],k-size[son[x][0]]-1);
	}
	inline int Split(int l,int r) 
	{
		int x=Find(root,l),y=Find(root,r+2);
		Splay(x,0); Splay(y,root); return lson(rson(root));
	}
	inline void Cover(int l,int r,int d) {int x=Split(l,r); Cov(x,d);}
	inline void Rever(int l,int r) {int x=Split(l,r); Rev(x);}
	inline void Inver(int l,int r) {int x=Split(l,r); Inv(x);}
	inline int Query(int l,int r) {int x=Split(l,r); return (lmax[x]+1)/2+(-rmin[x]+1)/2;}
}using namespace SplayTree; 

int main()
{
	Newnode(root,0,0);
	Newnode(son[root][1],root,0);
	
	N=read(),M=read();
	scanf("%s",s+1); 
	
	son[son[root][1]][0]=SplayTree::Build(1,N,son[root][1]);
	
	while (M--) {
		char opt[10]; scanf("%s",opt+1);
		int l=read(),r=read();
		switch (opt[1]) {
			case 'R' : scanf("%s",s+1); SplayTree::Cover(l,r,s[1]=='('? -1:1); break;
			case 'Q' : printf("%d\n",SplayTree::Query(l,r)); break;
			case 'S' : SplayTree::Rever(l,r); break;
			case 'I' : SplayTree::Inver(l,r); break;
		} 
	}
	
	return 0;
}

大過年的,寫啥Splay啊....這道題又捯飭了兩個多小時...真是自虐。

相關文章
相關標籤/搜索