二逼平衡树

Posted issue是fw

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了二逼平衡树相关的知识,希望对你有一定的参考价值。

#include <bits/stdc++.h>
using namespace std;
const int maxn = 1e5+10;
const int N = 4e7+10;
const int inf = 1e9;
#define mid (l+r>>1)
int n,m,a[maxn];

int id,root[maxn],ls[N],rs[N],sum[N],rt1[N],rt2[N];
int li[maxn],top;
int lowbit(int x){ return x&(-x); }
void get_point(int l,int r)
{
	rt1[0] = rt2[0] = 0;
	for(int i=l-1;i;i-=lowbit(i) )	rt1[++rt1[0]] = root[i];
	for(int i=r;i;i-=lowbit(i) )	rt2[++rt2[0]] = root[i];
}
void rt_left()
{
	for(int i=1;i<=rt1[0];i++)	rt1[i] = ls[rt1[i]];
	for(int i=1;i<=rt2[0];i++)	rt2[i] = ls[rt2[i]];
}
void rt_right()
{
	for(int i=1;i<=rt1[0];i++)	rt1[i] = rs[rt1[i]];
	for(int i=1;i<=rt2[0];i++)	rt2[i] = rs[rt2[i]];
}
void update_sgt(int &rt,int l,int r,int pos,int val)
{
	if( !rt )	rt = ++id;
	if( l==r )	{ sum[rt] += val; return; }
	if( pos<=mid )	update_sgt(ls[rt],l,mid,pos,val);
	else	update_sgt(rs[rt],mid+1,r,pos,val );
	sum[rt] = sum[ls[rt]]+sum[rs[rt]];
}
int ask_sgt_sum(int l,int r,int L,int R)
{
	if( l>R || r<L )	return 0;
	if( l>=L && r<=R )
	{
		int temp = 0;
		for(int i=1;i<=rt1[0];i++)	temp -= sum[rt1[i]];
		for(int i=1;i<=rt2[0];i++)	temp += sum[rt2[i]];
		return temp; 
	}
	int rt3[109],rt4[109];
	for(int i=0;i<=rt1[0];i++)	rt3[i] = rt1[i];
	for(int i=0;i<=rt2[0];i++)	rt4[i] = rt2[i];			
	rt_left();
	int x = ask_sgt_sum(l,mid,L,R);
	for(int i=1;i<=rt1[0];i++)	rt1[i] = rt3[i];
	for(int i=1;i<=rt2[0];i++)	rt2[i] = rt4[i];
	rt_right();
	return x+ask_sgt_sum(mid+1,r,L,R);	
}
int k_th(int l,int r,int k)
{
	if( l==r )
	{
		int temp = 0;
		for(int i=1;i<=rt1[0];i++)	temp -= sum[rt1[i]];
		for(int i=1;i<=rt2[0];i++)	temp += sum[rt2[i]];
		if( temp<k )	return inf;
		else	return l;		
	}
	int x = 0;
	for(int i=1;i<=rt1[0];i++)	x -= sum[ls[rt1[i]]];
	for(int i=1;i<=rt2[0];i++)	x += sum[ls[rt2[i]]];
	if(  k<=x )
	{
		rt_left();
		return k_th(l,mid,k);
	}
	else
	{
		rt_right();
		return k_th(mid+1,r,k-x );
	}
}
int prek_th(int l,int r,int k)
{	
	get_point(l,r);
	int kth = ask_sgt_sum(0,inf,0,k-1);//小于等于k-1的数字有多少个
	if( kth==0 )	return inf;
	get_point(l,r);
	int res = k_th(0,inf,kth);
	return res; 
}
int sufk_th(int l,int r,int k)
{
	get_point(l,r);
	int kth = ask_sgt_sum(0,inf,0,k)+1;
	get_point(l,r);
	int res = k_th( 0,inf,kth );
	return res;
}

void update_BIT(int x,int val,int zhi)//把x位置,在权值线段树上的val位置修改zhi  
{
	for(int i=x;i<=n;i+=lowbit(i) )	
		update_sgt( root[i],0,inf,val,zhi );
}
int main()
{
	cin >> n >> m;
	for(int i=1;i<=n;i++)	scanf("%d",&a[i] ),li[i] = a[i];
	
	for(int i=1;i<=n;i++)	update_BIT( i,a[i],1 );	
	for(int i=1;i<=m;i++)
	{
		int type,pos,l,r,k; scanf("%d",&type );
		if( type==1 )
		{
			scanf("%d%d%d",&l,&r,&k );
			get_point(l,r);//获取区间的权值线段树
			printf("%d\\n",ask_sgt_sum(0,inf,0,k-1)+1 );	
		}
		else if( type==2 )
		{
			scanf("%d%d%d",&l,&r,&k );
			get_point(l,r);//获取区间的权值线段树
			printf("%d\\n",k_th(0,inf,k) );			
		}
		else if( type==3 )
		{
			scanf("%d%d",&pos,&k );
			update_BIT( pos,a[pos],-1 );
			a[pos] = k;
			update_BIT( pos,a[pos],1 );
		}
		else if( type==4 )
		{
			scanf("%d%d%d",&l,&r,&k );
			int res = prek_th(l,r,k);
			if( res==inf )	printf("-2147483647\\n");
			else	printf("%d\\n",res );
		}
		else if( type==5 )
		{
			scanf("%d%d%d",&l,&r,&k );		
			int res = sufk_th(l,r,k);
			if( res==inf )	printf("2147483647\\n");
			else	printf("%d\\n",res );
		}
	}
}

以上是关于二逼平衡树的主要内容,如果未能解决你的问题,请参考以下文章

bzoj3196: Tyvj 1730 二逼平衡树

模板二逼平衡树(树套树)

LibreOJ #106. 二逼平衡树

BZOJ 3196 二逼平衡树

bzoj3196 Tyvj 1730 二逼平衡树

bzoj3196 二逼平衡树——线段树套平衡树