[BZOJ3223]文艺平衡树

以前看着非旋treap的题解写了一发,没有真正理解

最近补了一下splay,于是用来练手

因为是区间翻转,子树大小不变,所以pushup时儿子对父亲没有影响

因此splay($x$)之前只需一路从根pushdown到$x$即可,不需要pushdown儿子

由于pushdown时交换儿子破坏了二叉搜索树的性质,故splay($x$)应该改为splay($k$),即splay排名第几的节点而不是splay节点编号

#include<stdio.h>
int ch[100010][2],fa[100010],siz[100010],tag[100010],root,n;
int build(int l,int r){
	int mid=(l+r)>>1,ls=0,rs=0;
	if(l<mid)ls=build(l,mid-1);
	if(mid<r)rs=build(mid+1,r);
	siz[mid]=1;
	if(ls){
		siz[mid]+=siz[ls];
		ch[mid][0]=ls;
		fa[ls]=mid;
	}
	if(rs){
		siz[mid]+=siz[rs];
		ch[mid][1]=rs;
		fa[rs]=mid;
	}
	return mid;
}
void pushup(int x){
	siz[x]=siz[ch[x][0]]+siz[ch[x][1]]+1;
}
void rot(int x){
	int y,z,B,f;
	y=fa[x];
	z=fa[y];
	f=(ch[y][0]==x);
	if(root==y)root=x;
	B=ch[x][f];
	if(B)fa[B]=y;
	fa[x]=z;
	fa[y]=x;
	ch[x][f]=y;
	ch[y][f^1]=B;
	if(ch[z][f^1]==y)ch[z][f^1]=x;
	if(ch[z][f]==y)ch[z][f]=x;
	pushup(y);
	pushup(x);
}
void swap(int&a,int&b){a^=b^=a^=b;}
void pushdown(int x){
	if(tag[x]){
		swap(ch[x][0],ch[x][1]);
		if(ch[x][0])tag[ch[x][0]]^=1;
		if(ch[x][1])tag[ch[x][1]]^=1;
		tag[x]=0;
	}
}
void splay(int k){
	int x,y,z;
	x=root;
	while(1){
		pushdown(x);
		if(siz[ch[x][0]]>=k)
			x=ch[x][0];
		else if(siz[ch[x][0]]+1<k){
			k-=(siz[ch[x][0]]+1);
			x=ch[x][1];
		}else
			break;
	}
	while(x!=root){
		if(fa[x]==root)
			rot(x);
		else{
			y=fa[x];
			z=fa[y];
			if((y==ch[z][0]&&x==ch[y][0])||(y==ch[z][1]&&x==ch[y][1])){
				rot(y);
				rot(x);
			}else{
				rot(x);
				rot(x);
			}
		}
	}
}
void reverse(int l,int r){
	if(l==1){
		if(r==n)
			tag[root]^=1;
		else{
			splay(r+1);
			tag[ch[root][0]]^=1;
			splay(siz[ch[root][0]]-siz[ch[ch[root][0]][1]]);
		}
		return;
	}
	if(r==n){
		splay(l-1);
		tag[ch[root][1]]^=1;
		splay(siz[ch[root][0]]+siz[ch[ch[root][1]][0]]+2);
		return;
	}
	splay(l-1);
	int t=root;
	root=ch[root][1];
	splay(r-siz[ch[t][0]]);
	tag[ch[root][0]]^=1;
	root=fa[root];
	splay(siz[ch[root][0]]+siz[ch[ch[ch[root][1]][0]][0]]+2);
}
void dfs(int x){
	pushdown(x);
	if(ch[x][0])dfs(ch[x][0]);
	printf("%d ",x);
	if(ch[x][1])dfs(ch[x][1]);
}
int main(){
	int m,l,r;
	scanf("%d%d",&n,&m);
	root=build(1,n);
	while(m--){
		scanf("%d%d",&l,&r);
		reverse(l,r);
	}
	dfs(root);
}
时间: 2024-12-06 02:48:37

[BZOJ3223]文艺平衡树的相关文章

[BZOJ3223]文艺平衡树 无旋Treap

3223: Tyvj 1729 文艺平衡树 Time Limit: 10 Sec  Memory Limit: 128 MB Description 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作:翻转一个区间,例如原有序序列是5 4 3 2 1,翻转区间是[2,4]的话,结果是5 2 3 4 1 Input 第一行为n,m n表示初始序列有n个数,这个序列依次是(1,2--n-1,n)  m表示翻转操作次数接下来m行每行两个数[l,r] 数据保证 1<=l<

JZYZOJ1998 [bzoj3223] 文艺平衡树 splay 平衡树

http://172.20.6.3/Problem_Show.asp?id=1998 平衡树区间翻转的板子,重新写一遍,给自己码一个板子. 1 #include<iostream> 2 #include<cstdio> 3 #include<cstring> 4 #include<algorithm> 5 #include<cmath> 6 #include<cstdlib> 7 using namespace std; 8 cons

[bzoj3223]文艺平衡树[splay]

//BY HZWER 1 #include<iostream> 2 #include<cstdio> 3 4 using namespace std; 5 6 inline int read() 7 { 8 int x=0,f=1;char ch=getchar(); 9 while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();} 10 while(ch>='0'&&ch<='9'){x=x*10

bzoj3223 文艺平衡树 treap

题意:求多次区间反转后的序列 一眼splay,我用了可持久treap实现的. 速度也不错 2008ms 这么想想splay好像没什么用了(大雾@lct #include<iostream> #include<cstdio> #include<cstdlib> #define N 200005 #define mp make_pair #define pa pair<int,int> using namespace std; struct treap{int

[bzoj3223]文艺平衡树(splay区间反转模板)

解题关键:splay模板题. #include<cstdio> #include<cstring> #include<algorithm> #include<cstdlib> #include<iostream> #include<cmath> using namespace std; typedef long long ll; const int N = 100005; int ch[N][2],par[N],val[N],cnt[

BZOJ3223: Tyvj 1729 文艺平衡树 [splay]

3223: Tyvj 1729 文艺平衡树 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 3595  Solved: 2029[Submit][Status][Discuss] Description 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作:翻转一个区间,例如原有序序列是5 4 3 2 1,翻转区间是[2,4]的话,结果是5 2 3 4 1 Input 第一行为n,m n表示初始序列有n个数,这个序列依次

[BZOJ3223][tyvj1729]文艺平衡树

3223: Tyvj 1729 文艺平衡树 Time Limit: 10 Sec  Memory Limit: 128 MB Submit: 5356  Solved: 3163 [Submit][Status][Discuss] Description 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作:翻转一个区间,例如原有序序列是5 4 3 2 1,翻转区间是[2,4]的话,结果是5 2 3 4 1 Input 第一行为n,m n表示初始序列有n个数,这个序列

[题解]bzoj 3223 文艺平衡树

3223: Tyvj 1729 文艺平衡树 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 3884  Solved: 2235[Submit][Status][Discuss] Description 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作:翻转一个区间,例如原有序序列是5 4 3 2 1,翻转区间是[2,4]的话,结果是5 2 3 4 1 Input 第一行为n,m n表示初始序列有n个数,这个序列依次

3223: Tyvj 1729 文艺平衡树

3223: Tyvj 1729 文艺平衡树 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 3473  Solved: 1962[Submit][Status][Discuss] Description 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作:翻转一个区间,例如原有序序列是5 4 3 2 1,翻转区间是[2,4]的话,结果是5 2 3 4 1 Input 第一行为n,m n表示初始序列有n个数,这个序列依次