题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4441

题意:一个数列,三种操作:(1)插入:找到没在当前数列中的最小的正整数i,将其插在位置p之后,并将-i插入某个位置使得满足先进先出(i表示进,-i表示出),这个位置尽量靠右;(2)删除:删掉数字i以及-i;(3)询问:查询i和-i之间的数字的和。

思路:对于没在数列中的数字可以用一个set直接维护。i的插入是正常的splay操作。对于-i的插入,我们首先找到i之前有几个正数,比如有x个,那么-i必然是插在第x+1个负数的前面,因此在splay节点中要保存负数的个数。删除和查询都是splay正常操作。

struct node
{
	i64 sum;
	int val;
    int size[2];
	int Size;
    node *c[2],*p;

};

node a[N],*root,*nullNode;
int cnt;

void pushUp(node *p)
{
    if(p==nullNode) return;

	p->Size=1;
	p->sum=p->val;
	p->size[0]=p->size[1]=0;
	if(p->val>0) p->size[1]++;
	else if(p->val<0) p->size[0]++;

	if(p->c[0]!=nullNode)
	{
		p->Size+=p->c[0]->Size;
		p->sum+=p->c[0]->sum;
		p->size[0]+=p->c[0]->size[0];
		p->size[1]+=p->c[0]->size[1];
	}

	if(p->c[1]!=nullNode)
	{
		p->Size+=p->c[1]->Size;
		p->sum+=p->c[1]->sum;
		p->size[0]+=p->c[1]->size[0];
		p->size[1]+=p->c[1]->size[1];
	}
}

int ok(node *p)
{
    return p!=nullNode&&(p!=&a[1])&&(p!=&a[2]);
}

node *newNode(int val,node *p)
{
    node *e=&a[val>0?val:-val+100000];
    e->c[0]=e->c[1]=nullNode;
    e->p=p;
    e->Size=1;
	e->val=e->sum=val;
	e->size[0]=e->size[1]=0;

	if(val>0) e->size[1]++;
	else e->size[0]++;
    return e;
}

void init()
{
    nullNode=new node();
    nullNode->size[0]=nullNode->size[1]=0;
	nullNode->Size=0;
    nullNode->sum=0;
	nullNode->c[0]=nullNode->c[1]=nullNode->p=nullNode;

    root=new node();
	root->c[0]=root->c[1]=root->p=nullNode;
    root->sum=0;
	root->val=0;
	root->size[0]=root->size[1]=0;
	root->Size=1;

	root->c[1]=new node();
	root->c[1]->c[0]=root->c[1]->c[1]=nullNode;
	root->c[1]->p=root;
    root->c[1]->sum=root->c[1]->val=-1;
	root->c[1]->Size=1;
	root->c[1]->size[0]=root->c[1]->size[1]=0;
	root->c[1]->size[0]++;

    pushUp(root);

}

void zig(node *x)
{
    node *p=x->p,*q=p->p;
    p->c[0]=x->c[1];
    if(x->c[1]!=nullNode) x->c[1]->p=p;
    x->c[1]=p;
    p->p=x;
    x->p=q;
    if(q!=nullNode)
    {
        if(q->c[0]==p) q->c[0]=x;
        else q->c[1]=x;
    }
    pushUp(p);
    pushUp(x);
    if(root==p) root=x;
}

void zag(node *x)
{
    node *p=x->p,*q=p->p;
    p->c[1]=x->c[0];
    if(x->c[0]!=nullNode) x->c[0]->p=p;
    x->c[0]=p;
    p->p=x;
    x->p=q;
    if(q!=nullNode)
    {
        if(q->c[0]==p) q->c[0]=x;
        else q->c[1]=x;
    }
    pushUp(p);
    pushUp(x);
    if(root==p) root=x;
}

void splay(node *x,node *goal)
{
    while(x->p!=goal)
    {
        if(x->p->p!=goal)
        {
            if(x->p->p->c[0]==x->p)
            {
                if(x->p->c[0]==x) zig(x->p),zig(x);
                else zag(x),zig(x);
            }
            else
            {
                if(x->p->c[1]==x) zag(x->p),zag(x);
                else zig(x),zag(x);
            }
        }
        else
        {
            if(x->p->c[0]==x) zig(x);
            else zag(x);
        }
    }
}

void select(int k,node *goal)
{
    node *p=root;
    while(k!=p->c[0]->Size+1)
    {
        if(k<=p->c[0]->Size) p=p->c[0];
        else
        {
            k-=1+p->c[0]->Size;
            p=p->c[1];
        }
    }
	splay(p,goal);
}

int curNodeNum;

void insert(int p,int val)
{
    select(p,nullNode);
    select(p+1,root);
    node *e=newNode(val,root->c[1]);
    root->c[1]->c[0]=e;
    splay(e,nullNode);

	curNodeNum++;
}

node *getPre(node *u)
{
	splay(u,nullNode);
	u=u->c[0];
	while(u->c[1]!=nullNode) u=u->c[1];
	return u;
}
node *getNext(node *u)
{
	splay(u,nullNode);
	u=u->c[1];
	while(u->c[0]!=nullNode) u=u->c[0];
	return u;
}

node *getKth0(int k)
{
    node *p=root;
    while(1)
    {
        if(p->c[0]->size[0]>=k) p=p->c[0];
        else if(p->c[0]->size[0]+1==k)
        {
            if(p->val<0) return p;
            k-=p->c[0]->size[0];
            p=p->c[1];
        }
        else
        {
            k-=p->c[0]->size[0];
            if(p->val<0) k--;
            p=p->c[1];
        }
    }
}

void del(node *p)
{
	node *u=getPre(p);
	node *v=getNext(p);
	splay(u,nullNode);
	splay(v,root);
	root->c[1]->c[0]=nullNode;
	splay(root->c[1],nullNode);

	curNodeNum--;
}

set<int> S;
int n;

void deal()
{
	S.clear();
	init();

	curNodeNum=2;

	int i;
	for(i=1;i<=100000;i++) S.insert(i);
	while(n--)
	{
		char op[10];
		int p;
		scanf("%s%d",op,&p);
		if(op[0]=='i')
		{
			int t=*S.begin();
			S.erase(t);
			insert(p+1,t);

			splay(a+t,nullNode);
			int num=0;
			if(root->c[0]!=nullNode) num+=root->c[0]->size[1];
			if(root->val>0) num++;

			node *u=getKth0(num);

			node *v=getPre(u);
			splay(v,nullNode);
			splay(u,root);
			node *tmp=newNode(-t,root->c[1]);
			root->c[1]->c[0]=tmp;
			splay(tmp,nullNode);

			curNodeNum++;

		}
		else if(op[0]=='r')
		{
			S.insert(p);
			del(a+p);
			del(a+p+100000);
		}
		else
		{
			node *u=a+p;
			node *v=a+p+100000;
			splay(u,nullNode);
			splay(v,root);
			i64 sum=root->c[1]->c[0]->sum;
			output(sum); puts("");
		}
	}
}

int main()
{
	int num=0;
	while(scanf("%d",&n)!=-1)
	{
		printf("Case #%d:\n",++num);
		deal();
	}
}

HDU 4441 Queue Sequence(splay)的更多相关文章

  1. HDU 4441 Queue Sequence(优先队列+Treap树)(2012 Asia Tianjin Regional Contest)

    Problem Description There's a queue obeying the first in first out rule. Each time you can either pu ...

  2. HDU 1711 Number Sequence(数列)

    HDU 1711 Number Sequence(数列) Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Ja ...

  3. HDU 1005 Number Sequence(数列)

    HDU 1005 Number Sequence(数列) Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Jav ...

  4. hdu 4915 Parenthese sequence(模拟)2014多培训学校5现场

    Parenthese sequence                                                                     Time Limit: ...

  5. hdu 6047 Maximum Sequence(贪心)

    Description Steph is extremely obsessed with "sequence problems" that are usually seen on ...

  6. HDU 4441 Queue Sequence

    http://acm.hdu.edu.cn/showproblem.php?pid=4441 题意:对于一个序列,每次有三种操作   insert pos  表示在pos插入一个数,这个数是最小的正数 ...

  7. HDU 1711 Number Sequence(KMP)附带KMP的详解

    题目代号:HDU 1711 题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1711 Number Sequence Time Limit: 10000/ ...

  8. hdu 6299 Balanced Sequence (贪心)

    Balanced Sequence Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others ...

  9. HDU 1890 Robotic Sort(splay)

    [题目链接] http://acm.hdu.edu.cn/showproblem.php?pid=1890 [题意] 给定一个序列,每次将i..P[i]反转,然后输出P[i],P[i]定义为当前数字i ...

随机推荐

  1. android 三种定位方式

    http://www.cnblogs.com/oudi/archive/2012/03/22/2411509.html 最近在看android关于定位的方式,查了很多资料,也做了相关实验,在手机上做了 ...

  2. zw版【转发·台湾nvp系列例程】HALCON EquHistoImage(Delphi)

    zw版[转发·台湾nvp系列例程]HALCON EquHistoImage(Delphi) zw版[转发·台湾nvp系列例程]HALCON EquHistoImage(Delphi) (Delphi ...

  3. [CrunchBang]禁止“桌面上鼠标滚轮切换工作区桌面“

    鼠标滚轮切换虚拟桌面相关问题, 编辑 ~/.config/openbox/rc.xml 在   <context name="Desktop">段: <mouse ...

  4. 【sinatra】结合Padrino framework

    用Sinatra来做复杂的Web应用时,会出现若干个比较麻烦的点. 要手工作成一个个的应用骨架.作成test .view.public目录等等 将Sinatra DSL集中在一个类中的话,画面数量增加 ...

  5. [翻译] 使用ASP.NET MVC操作过滤器记录日志

    [翻译] 使用ASP.NET MVC操作过滤器记录日志 原文地址:http://www.singingeels.com/Articles/Logging_with_ASPNET_MVC_Action_ ...

  6. php socket函数详解

    最近在用socket实现服务端向客户端主动推送消息 函数名 描述 socket_accept() 接受一个Socket连接 socket_bind() 把socket绑定在一个IP地址和端口上 soc ...

  7. C++中new和delete来创建和释放动态数组

    在C++编程中,使用new创建数组然后用delete来释放. 一.创建并释放一维数组 #include<iostream> using namespace std; int main() ...

  8. 【转】Program Files (x86)文件夹是干什么的

    转载地址:http://leonax.net/p/2373/what-program-files-x86-folder-for/ 今天在微博上看到有人问这个问题.在装好64位的Windows之后,能在 ...

  9. 【转】使用JDK自带jvisualvm监控tomcat

    转载地址: http://my.oschina.net/kone/blog/157239 jdk自带有个jvisualvm工具.该工具是用来监控java运行程序的cpu.内存.线程等的使用情况.并且使 ...

  10. hdu1074 Doing Homework

    这题比较有意思,暴力搜索必然tle,可以用状态压缩dp解决. 我们先不考虑完成所有作业的扣分,而考虑其一个子集的情况. 假设我们得到了完成某子集S对应的作业最少扣分,我们试着向该子集中增加一个元素a, ...