一眼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
#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啊....這道題又捯飭了兩個多小時...真是自虐。