【题解】数列

前置芝士 Splay

[题目链接](https://www.luogu.org/problem/P2710)

这种东西,就要上我们的序列之王——\(Splay\)了。

相比\(NOI2005\)维护序列来说,仅仅多了一个单点查询。

先说一下\(Splay\)维护区间信息的芝士:

我要修改区间\([l,r]\)时,我们要操作的编号,应该是在\(Splay\)树上对应的\([l,r]\).所以,可以写一个查询排名位置的函数来完成这个操作。

考虑如何把一个区间弄到一起。如果我们要包含区间端点,显然我们应该操作的是\([l-1,r+1]\)而不是\([l,r]\)。我们将区间端点同时\(+1\),就得到\([l,r+2]\).

所以我们对区间修改的时候,先把\(Kth(l)\)转到\(root\),再把\(Kth(r+2)\)转到\(Kth(l)\)的右子树。那么,\(Kth(r+2)\)的左子树就是这个区间了。

然后考虑题目中的操作。

\(Insert\)

嗯,笔者直接暴力依次插入的。其实还有更好的办法,笔者学习中,有时间会补。把每一个节点插入到\(x+1\)的位置就好了。

\(Delete\)

这个比较简单,直接把上面的操作区间(注意是我们的操作区间而不是题目给的编号)和上面一样操作,之后把\(R\)的左子树删掉就行了。

下面,均以\(L=Kth(l),R=Kth(r+2)\)

\(Reverse\)

\(Splay\)基本操作,直接对要修改的区间打上标记,维护一下\(pushup\)\(pushdown\)即可。本题维护的标记很多,注意一下。

这里修改的区间也像上面一样,转到根在打标记。

\(MAKE-SAME\)

区间覆盖,打上标记即可,注意这个标记的优先级比\(rev\)高,因为区间都覆盖了,值都一样就没必要翻转了。

\(GET-SUM\)

树上维护区间和\(sum\)即可。

\(Get\)

单点查询,直接输出\(tr[Kth(x+1)].val\)即可。

\(MAX-SUM\)

区间最大子段和,带修改,本题难点。

这里学习一下\(nlogn\)的方法。

树上维护\(ls,rs,ms\),分别表示前缀最大段和,后缀最大段和以及本区间的最大子段和。

那么我们可以维护它们了。具体看代码吧。

#include<iostream>
#include<cstring>
#include<cstdio>
#include<string>
using namespace std;
const int inf=2147483647;
int n,m,id,rt;
inline int read(){
	int s=0,w=1;
	char ch=getchar();
	while(ch<'0'||ch>'9'){
		if(ch=='-')w=-1;
		ch=getchar();
	}
	while(ch>='0'&&ch<='9'){
		s=(s<<1)+(s<<3)+(ch^48);
		ch=getchar();
	}
	return s*w;
}
char opt[20];
struct node{
	int ch[2],val,sum,rev;
	int tag,fa,ls,ms,rs,siz;
}tr[500010];
inline void swap(int &x,int &y){x^=y^=x^=y;}
inline void pushup(int x){
	int lc=tr[x].ch[0],rc=tr[x].ch[1];
	tr[x].sum=tr[x].val+tr[lc].sum+tr[rc].sum;
	tr[x].siz=tr[lc].siz+tr[rc].siz+1;
	tr[x].ls=max(tr[lc].ls,tr[lc].sum+tr[x].val+tr[rc].ls);//注意子段和的维护 
	tr[x].rs=max(tr[rc].rs,tr[lc].rs+tr[rc].sum+tr[x].val);//可以自己理解一下,比较难描述 
	tr[x].ms=max(tr[lc].ms,max(tr[rc].ms,tr[lc].rs+tr[x].val+tr[rc].ls));
	//前缀就等于左孩子的前缀 和左孩子的全部加上x的值加上右孩子的前缀的max 
	//后缀就等于右孩子的后缀 和左孩子的后缀加上x的值加上右孩子的全部的max
	//上面的全部即sum
	//那么区间最大子段和就是代码上所说的
	//等于左孩子的最大子段和,右孩子的最大子段和,左孩子后缀加x的值加右孩子前缀的最大值。 
} 
inline void pushdown(int x){
	int lc=tr[x].ch[0],rc=tr[x].ch[1];
	if(tr[x].tag!=-inf){
		int p=tr[x].tag;
		if(lc){
			tr[lc].val=tr[lc].tag=p;
			tr[lc].sum=p*tr[lc].siz;
			if(p>=0)tr[lc].ls=tr[lc].rs=tr[lc].ms=tr[lc].sum;
			else tr[lc].ls=tr[lc].rs=0,tr[lc].ms=p;
		}
		if(rc){
			tr[rc].val=tr[rc].tag=p;
			tr[rc].sum=p*tr[rc].siz;
			if(p>=0)tr[rc].ls=tr[rc].rs=tr[rc].ms=tr[rc].sum;
			else tr[rc].ls=tr[rc].rs=0,tr[rc].ms=p;
		}
		tr[x].tag=-inf,tr[x].rev=0;
	}
	if(tr[x].rev){
		tr[x].rev=0;
		tr[lc].rev^=1;
		tr[rc].rev^=1;
		swap(tr[lc].ls,tr[lc].rs);
		swap(tr[rc].ls,tr[rc].rs);
		swap(tr[lc].ch[0],tr[lc].ch[1]);
		swap(tr[rc].ch[0],tr[rc].ch[1]);
	}
}
inline void rotate(int x){
	int y=tr[x].fa,z=tr[y].fa,k=tr[y].ch[1]==x;
	tr[z].ch[tr[z].ch[1]==y]=x;tr[x].fa=z;
	tr[y].ch[k]=tr[x].ch[k^1];tr[tr[x].ch[k^1]].fa=y;
	tr[x].ch[k^1]=y;tr[y].fa=x;pushup(y);pushup(x);
}
inline void splay(int x,int g){
	while(tr[x].fa!=g){
		int y=tr[x].fa,z=tr[y].fa;
		if(z!=g)
			(tr[y].ch[0]==x)^(tr[z].ch[0]==y)?rotate(x):rotate(y);
		rotate(x);
	}
	if(!g)rt=x;
}
inline int Kth(int x){
	int u=rt;
	if(!x)return 0;
	while(u){
		pushdown(u);
		int y=tr[u].ch[0];
		if(tr[y].siz>=x)u=y;
		else{
			x-=tr[y].siz+1;
			if(!x)return u;
			u=tr[u].ch[1];
		}
	}
	return 0;
}
inline void Ins(int x,int val){
	int F=Kth(x);splay(F,0);
	int p=++id;
	tr[p].siz=1;tr[p].val=tr[p].sum=val;
	tr[p].ms=val;tr[p].rev=0;tr[p].tag=-inf;
	tr[p].fa=F;tr[p].siz=1;
	if(val>=0)tr[p].ls=tr[p].rs=val;
	if(F){
		tr[p].ch[1]=tr[F].ch[1];
		tr[tr[F].ch[1]].fa=p;
		tr[F].ch[1]=p;pushup(p);
		pushup(F);
	}
	splay(p,0);
}
inline void change(int x,int y){
	int L=Kth(x),R=Kth(y+2);
	splay(L,0);splay(R,L);
	int g=tr[R].ch[0];
	if(tr[g].tag==-inf){
		tr[g].rev^=1;
		swap(tr[g].ls,tr[g].rs);
		swap(tr[g].ch[0],tr[g].ch[1]);
	}
	pushup(R);pushup(L);
}
inline int Get(int x,int y){
	int L=Kth(x),R=Kth(y+2);
	splay(L,0);splay(R,L);
	return tr[tr[R].ch[0]].sum;
}
inline void Del(int x,int y){
	int L=Kth(x),R=Kth(y+2);
	splay(L,0);splay(R,L);
	tr[R].ch[0]=0;
	pushup(R);pushup(L);
}
inline void Make(int x,int y,int z){
	int L=Kth(x),R=Kth(y+2);
	splay(L,0);splay(R,L);
	int g=tr[R].ch[0];
	tr[g].val=z;tr[g].tag=z;
	tr[g].sum=tr[g].siz*z;
	if(z>=0)tr[g].ls=tr[g].rs=tr[g].ms=tr[g].sum;
	else tr[g].ls=tr[g].rs=0,tr[g].ms=z;
	pushup(R);pushup(L);
}
inline int Get_pos(int x){
	return tr[Kth(x+1)].val;
}
int main(){
	tr[0].ms=-99999999;
	n=read(),m=read();
	Ins(0,-99999999);
	for(int i=1,x;i<=n;++i){
		scanf("%d",&x);
		Ins(i,x);
	}
	int x,y,l,r,tot;
	Ins(n+1,-99999999);
	while(m--){
		cin>>opt;
		if(opt[0]=='I'){
			x=read(),tot=read();
			for(int i=1;i<=tot;++i){
				x++;
				y=read();
				Ins(x,y);
			}
		}
		else if(opt[0]=='R'){
			l=read(),r=read();
			r=l+r-1;
			change(l,r);
		}
		else if(opt[0]=='G'){
			if(strlen(opt)>4){
				l=read(),r=read();
				r=l+r-1;
				if(r<l){
					printf("0\n");
					continue;
				}
				else printf("%d\n",Get(l,r));
			}
			else{
				x=read();
				printf("%d\n",Get_pos(x));
			}
		}
		else if(opt[0]=='D'){
			l=read(),r=read();
			r=l+r-1;
			Del(l,r);
		}
		else if(opt[0]=='M'){
			if(opt[2]=='K'){
				l=read(),r=read(),x=read();
				r=l+r-1;
				Make(l,r,x);
				continue;
			}
			else{
				l=read(),r=read();
				r=l+r-1;
				int L=Kth(l),R=Kth(r+2);
				splay(L,0);splay(R,L);
				cout<<tr[tr[R].ch[0]].ms<<endl;
			}
		}
	}
	return 0;
}
posted @ 2019-09-13 11:30  Refined_heart  阅读(232)  评论(0编辑  收藏  举报