【BZOJ3685】【zkw权值线段树】普通van Emde Boas树

时间:2023-03-08 17:25:24
【BZOJ3685】【zkw权值线段树】普通van Emde Boas树

原题传送门

因为马上要开始搞树套树了,所以学了一波权值线段树。。。毕竟是会点zkw线段树的,所以zkw线段树大法好!

解题思路:

  介绍一下权值线段树吧,其实感觉就是线段树的本义,就是你用线段树维护了数轴上的一段区间,然后统计这个区间内被覆盖的点的数量(我们认为同一个点可以被覆盖多次)

  然后这题你就直接权值线段树(因为0还挺麻烦的,所以我们暴力加个1)暴力搞一波即可,具体参见代码吧,不多赘述了。。。。

#include <stdio.h>
#define r register
#define getchar() (S==TT&&(TT=(S=BB)+fread(BB,1,1<<15,stdin),TT==S)?EOF:*S++)
char BB[<<],*S=BB,*TT=BB;
inline int in(){
r int x=;r char c;for (;(c=getchar())<''||c>'';);
for (x=c-'';(c=getchar())>=''&&c<='';x=(x<<)+(x<<)+c-'');
return x;
}bool b[<<];
int t[<<|],M,n,q;
inline void A(int k,int v){for (k+=M; k; k>>=)t[k]+=v;}
inline int Get_Min(int k){for (;k<=M;k=t[k<<]?k<<:k<<|);return k-M-;}
inline int Get_Max(int k){for (;k<=M;k=t[k<<|]?k<<|:k<<);return k-M-;}
inline int prefix(int v){
for (v+=M;v!=;v>>=)
if ((v&)&&t[v^]) return Get_Max(v^);
return -;
}
inline int suffix(int v){
for (v+=M;v!=;v>>=)
if ((~v&)&&t[v^]) return Get_Min(v^);
return -;
}
int main(){
for(n=in(),q=in(),M=; M<=n; M<<=);while(q--){
r int op=in(),x;
if (op==){if (!b[x=in()+]) b[x]=,A(x,);}else
if (op==){if (b[x=in()+]) b[x]=,A(x,-);}else
if (op==) printf("%d\n",t[]?Get_Min():-);else
if (op==) printf("%d\n",t[]?Get_Max():-);else
if (op==) printf("%d\n",prefix(in()+));else
if (op==) printf("%d\n",suffix(in()+));else
printf("%d\n",(b[x=in()+]?:-));
}
}