zoukankan      html  css  js  c++  java
  • 【BZOJ1500】[NOI2005]维修数列 Splay

    【BZOJ1500】[NOI2005]维修数列

    Description

    Input

    输入的第1 行包含两个数N 和M(M ≤20 000),N 表示初始时数列中数的个数,M表示要进行的操作数目。
    第2行包含N个数字,描述初始时的数列。
    以下M行,每行一条命令,格式参见问题描述中的表格。
    任何时刻数列中最多含有500 000个数,数列中任何一个数字均在[-1 000, 1 000]内。
    插入的数字总数不超过4 000 000个,输入文件大小不超过20MBytes。

    Output

    对于输入数据中的GET-SUM和MAX-SUM操作,向输出文件依次打印结果,每个答案(数字)占一行。

    Sample Input

    9 8
    2 -6 3 5 1 -5 -3 6 3
    GET-SUM 5 4
    MAX-SUM
    INSERT 8 3 -5 7 2
    DELETE 12 1
    MAKE-SAME 3 3 2
    REVERSE 3 6
    GET-SUM 5 4
    MAX-SUM

    Sample Output

    -1
    10
    1
    10

    HINT

    题解:裸的Splay,只不过全是细节,我还是不够细啊~

    *要开滚动数组记录有哪些空余的位置,防止MLE(但实测600000的数组就行)

    *INSERT:直接一个一个往里加就行,加成一条链也无所谓,不会TLE

    *DELETE:因为要释放空间,所以必须一个一个删除,递归即可

    *MAKE_SAME:没啥说的
    *REVERSE:最好是先修改,再给儿子打标记(也就是说标记只对儿子起作用),这样比较清晰

    *GET_SUM:没啥说的
    *MAX_SUM:巨恶心的pushup和pushdown,看代码就知道了

    #include <cstdio>
    #include <cstring>
    #include <iostream>
    #include <queue>
    using namespace std;
    struct point
    {
    	int ch[2],fa,siz,tag,v,sv,sm,ls,rs,re;
    }s[600010];
    int n,m,root;
    int num[500010];
    char str[20];
    queue<int> q;
    void pushdown(int x)
    {
    	if(s[x].re)
    	{
    		swap(s[s[x].ch[0]].ch[0],s[s[x].ch[0]].ch[1]);
    		swap(s[s[x].ch[1]].ch[0],s[s[x].ch[1]].ch[1]);
    		swap(s[s[x].ch[0]].ls,s[s[x].ch[0]].rs);
    		swap(s[s[x].ch[1]].ls,s[s[x].ch[1]].rs);
    		if(s[x].ch[0])	s[s[x].ch[0]].re^=1;
    		if(s[x].ch[1])	s[s[x].ch[1]].re^=1;
    		s[x].re=0;
    	}
    	if(s[x].tag!=1<<30)
    	{
    		if(s[x].ch[0])
    		{
    			s[s[x].ch[0]].v=s[s[x].ch[0]].tag=s[x].tag;
    			s[s[x].ch[0]].sv=s[x].tag*s[s[x].ch[0]].siz;
    			s[s[x].ch[0]].ls=s[s[x].ch[0]].rs=max(s[s[x].ch[0]].sv,0);
    			s[s[x].ch[0]].sm=max(s[s[x].ch[0]].sv,s[x].tag);
    		}
    		if(s[x].ch[1])
    		{
    			s[s[x].ch[1]].v=s[s[x].ch[1]].tag=s[x].tag;
    			s[s[x].ch[1]].sv=s[x].tag*s[s[x].ch[1]].siz;
    			s[s[x].ch[1]].ls=s[s[x].ch[1]].rs=max(s[s[x].ch[1]].sv,0);
    			s[s[x].ch[1]].sm=max(s[s[x].ch[1]].sv,s[x].tag);
    		}
    		s[x].tag=1<<30;
    	}
    }
    void pushup(int x)
    {
    	s[x].siz=s[s[x].ch[0]].siz+s[s[x].ch[1]].siz+1;
    	s[x].ls=max(s[s[x].ch[0]].ls,s[s[x].ch[0]].sv+s[x].v+s[s[x].ch[1]].ls);
    	s[x].rs=max(s[s[x].ch[1]].rs,s[s[x].ch[1]].sv+s[x].v+s[s[x].ch[0]].rs);
    	s[x].sm=max(s[s[x].ch[0]].rs+s[x].v+s[s[x].ch[1]].ls,max(s[s[x].ch[0]].sm,s[s[x].ch[1]].sm));
    	s[x].sv=s[s[x].ch[0]].sv+s[x].v+s[s[x].ch[1]].sv;
    }
    int readin()
    {
    	int ret=0,f=1;	char gc=getchar();
    	while(gc<'0'||gc>'9'){if(gc=='-')f=-f;	gc=getchar();}
    	while(gc>='0'&&gc<='9')	ret=ret*10+gc-'0',gc=getchar();
    	return ret*f;
    }
    int rotate(int x,int &k)
    {
    	int y=s[x].fa,z=s[y].fa,d=(x==s[y].ch[1]);
    	if(z)	s[z].ch[y==s[z].ch[1]]=x;
    	if(y==k)	k=x;
    	s[x].fa=z,s[y].fa=x,s[y].ch[d]=s[x].ch[d^1];
    	if(s[x].ch[d^1])	s[s[x].ch[d^1]].fa=y;
    	s[x].ch[d^1]=y;
    	pushup(y),pushup(x);
    }
    void splay(int x,int &k)
    {
    	while(x!=k)
    	{
    		int y=s[x].fa,z=s[y].fa;
    		if(y!=k)
    		{
    			if((x==s[y].ch[1])^(y==s[z].ch[1]))	rotate(x,k);
    			else	rotate(y,k);
    		}
    		rotate(x,k);
    	}
    }
    void build(int l,int r,int last)
    {
    	if(l>r)	return ;
    	int mid=l+r>>1;
    	s[mid].fa=last,s[last].ch[mid>last]=mid;
    	s[mid].v=num[mid];
    	s[mid].tag=1<<30;
    	build(l,mid-1,mid),build(mid+1,r,mid);
    	pushup(mid);
    }
    int find(int x,int y)
    {
    	pushdown(x);
    	if(s[s[x].ch[0]].siz+1==y)	return x;
    	if(y<=s[s[x].ch[0]].siz)	return find(s[x].ch[0],y);
    	return find(s[x].ch[1],y-s[s[x].ch[0]].siz-1);
    }
    void del(int &x)
    {
    	if(!x) return;
    	q.push(x);
    	del(s[x].ch[0]),del(s[x].ch[1]);
    	x=0;
    }
    int main()
    {
    	n=readin(),m=readin();
    	int i,j,a,b,c,t,u;
    	s[0].sm=num[1]=num[n+2]=-1<<30;
    	for(i=1;i<=n;i++)	num[i+1]=readin();
    	n+=2,root=(n+1)/2;
    	build(1,root-1,root),build(root+1,n,root);
    	s[root].v=num[root];
    	s[root].tag=1<<30;
    	pushup(root);
    	for(i=n+1;i<=600000;i++)	q.push(i);
    	for(i=1;i<=m;i++)
    	{
    		scanf("%s",str);
    		switch(str[2])
    		{
    			case 'S':
    			{
    				a=readin()+1,b=readin();
    				splay(find(root,a+1),root),splay(find(root,a),s[root].ch[0]);
    				t=s[root].ch[0];
    				for(j=1;j<=b;j++)
    				{
    					c=readin();
    					u=q.front(),q.pop();
    					s[u].v=c;
    					s[t].ch[1]=u,s[u].fa=t;
    					s[u].re=0,s[u].tag=1<<30;
    					t=u;
    				}
    				while(t!=root)
    				{
    					pushup(t);
    					t=s[t].fa;
    				}
    				break;
    			}
    			case 'L':
    			{
    				a=readin()+1,b=readin();
    				splay(find(root,a-1),root),splay(find(root,a+b),s[root].ch[1]);
    				del(s[s[root].ch[1]].ch[0]);
    				pushup(s[root].ch[1]);
    				break;
    			}
    			case 'K':
    			{
    				a=readin()+1,b=readin();
    				splay(find(root,a-1),root),splay(find(root,a+b),s[root].ch[1]);
    				c=s[s[root].ch[1]].ch[0];
    				s[c].v=s[c].tag=readin();
    				s[c].re=0;
    				s[c].sv=s[c].siz*s[c].v;
    				s[c].ls=s[c].rs=max(s[c].sv,0);
    				s[c].sm=max(s[c].sv,s[c].v);
    				break;
    			}
    			case 'V':
    			{
    				a=readin()+1,b=readin();
    				splay(find(root,a-1),root),splay(find(root,a+b),s[root].ch[1]);
    				c=s[s[root].ch[1]].ch[0];
    				swap(s[c].ch[0],s[c].ch[1]);
    				swap(s[c].ls,s[c].rs);
    				s[c].re=1;
    				break;
    			}
    			case 'T':
    			{
    				a=readin()+1,b=readin();
    				splay(find(root,a-1),root),splay(find(root,a+b),s[root].ch[1]);
    				printf("%d
    ",s[s[s[root].ch[1]].ch[0]].sv);
    				break;
    			}
    			case 'X':
    			{
    				printf("%d
    ",s[root].sm);
    				break;
    			}
    		}
    	}
    	return 0;
    }
  • 相关阅读:
    TOI2008 二元一次联立方程式
    福州三中信息学奥赛培训网址
    Youtube视频下载方式
    abs和其他绝对值的区别
    【转】C语言浮点数运算
    [转]解析字符串的方法
    TOI2008 大数运算
    【ZeroJudge】d781 Anagram
    【转载】NDatabase 5 Minutes Tutorial
    左岸读书编程是最能表达人类的思维的语言
  • 原文地址:https://www.cnblogs.com/CQzhangyu/p/6440275.html
Copyright © 2011-2022 走看看