struct node{
int l,r;//左右儿子
int sd,sz;//sd为节点所在子树的实际大小,而sz则为包括所有删除的元素在内的子树大小
int cnt,val;//cnt为数量,val为键值
};
inline void update(int p)
{
f[p].sd=f[f[p].l].sd+f[f[p].r].sd+f[p].cnt;
f[p].sz=f[f[p].l].sz+f[f[p].r].sz+f[p].cnt;
}
inline bool Can_Rbu(int p)
{
return f[p].cnt&&(f[p].sz*alpha<=(double)max(f[f[p].l].sz,f[f[p].r].sz)||f[p].sz*alpha>=(double)f[p].sd);
}
int tmp[100010];
void Rbu_Flatten(int &num,int p)
{
if(!p) return;//空儿子返回
Rbu_Flatten(num,f[p].l);//搜左儿子
if(f[p].cnt) tmp[num++]=p;//加入当前节点
Rbu_Flatten(num,f[p].r);//搜右儿子
}
int Rbu_Build(int l,int r)
{
if(l>=r) return 0;//二分边界
int mid=(l+r)>>1;
f[tmp[mid]].l=Rbu_Build(l,mid);//建左儿子
f[tmp[mid]].r=Rbu_Build(mid+1,r);//建右儿子
update(tmp[mid]);//更新信息
return tmp[mid];
}
void Rbu(int &p)
{
int x=0;
Rbu_Flatten(x,p);
p=Rbu_Build(0,x);//这里注意下标从0开始可以节省很多事情
}
void Insert(int &p,int val)
{
if(!p)
{
p=++tot;
f[p].cnt=f[p].sd=f[p].sz=1;
f[p].val=val;
f[p].l=f[p].r=0;
}
else
{
if(f[p].val==val) f[p].cnt++;
else if(val<f[p].val) Insert(f[p].l,val);
else Insert(f[p].r,val);
update(p);
if(Can_Rbu(p)) Rbu(p);
}
}
void del(int &p,int val)
{
if(!p) return;
f[p].sd--;
if(f[p].val==val)
{
if(f[p].cnt) f[p].cnt--;
}
else
{
if(val<f[p].val) del(f[p].l,val);
else del(f[p].r,val);
update(p);
}
if(Can_Rbu(p)) Rbu(p);
}
int GetVal(int p,int rk)
{
if(!p) return 0;
if(rk>f[f[p].l].sd&&rk<=f[f[p].l].sd+f[p].cnt) return f[p].val;
else if(rk>f[f[p].l].sd+f[p].cnt) return GetVal(f[p].r,rk-f[f[p].l].sd-f[p].cnt);
else return GetVal(f[p].l,rk);
}
int Get_uprb(int p,int val)
{
if(!p) return 1;
if(f[p].val==val&&f[p].cnt) return f[f[p].l].sd+f[p].cnt+1;
else if(val<f[p].val) return Get_uprb(f[p].l,val);
else return f[f[p].l].sd+f[p].cnt+Get_uprb(f[p].r,val);
}
int Get_lwrb(int p,int val)
{
if(!p) return 0;
if(f[p].val==val&&f[p].cnt) return f[f[p].l].sd;
else if(val<f[p].val) return Get_lwrb(f[p].l,val);
else return f[f[p].l].sd+f[p].cnt+Get_lwrb(f[p].r,val);
}
#include<bits/stdc++.h>
using namespace std;
template <typename T>
void read(T &x) {
x = 0;
int f = 1;
char ch = getchar();
while (!isdigit(ch)) {
if (ch == '-') f = -1;
ch = getchar();
}
while (isdigit(ch)) {
x = x * 10 + (ch ^ 48);
ch = getchar();
}
x *= f;
return;
}
template <typename T>
void write(T x)
{
if(x < 0) {
putchar('-');
x = -x;
}
if(x > 9)
write(x/10);
putchar(x % 10 + '0');
return;
}
int n;
int rt,tot=0;
struct node{
int l,r;
int sd,sz;
int cnt,val;
}f[100010];
const double alpha=0.75;
inline void update(int p)
{
f[p].sd=f[f[p].l].sd+f[f[p].r].sd+f[p].cnt;
f[p].sz=f[f[p].l].sz+f[f[p].r].sz+f[p].cnt;
}
inline bool Can_Rbu(int p)
{
return f[p].cnt&&(f[p].sz*alpha<=(double)max(f[f[p].l].sz,f[f[p].r].sz)||f[p].sz*alpha>=(double)f[p].sd);
}
int tmp[100010];
void Rbu_Flatten(int &num,int p)
{
if(!p) return;
Rbu_Flatten(num,f[p].l);
if(f[p].cnt) tmp[num++]=p;
Rbu_Flatten(num,f[p].r);
}
int Rbu_Build(int l,int r)
{
if(l>=r) return 0;
int mid=(l+r)>>1;
f[tmp[mid]].l=Rbu_Build(l,mid);
f[tmp[mid]].r=Rbu_Build(mid+1,r);
update(tmp[mid]);
return tmp[mid];
}
void Rbu(int &p)
{
int x=0;
Rbu_Flatten(x,p);
p=Rbu_Build(0,x);
}
void Insert(int &p,int val)
{
if(!p)
{
p=++tot;
f[p].cnt=f[p].sd=f[p].sz=1;
f[p].val=val;
f[p].l=f[p].r=0;
}
else
{
if(f[p].val==val) f[p].cnt++;
else if(val<f[p].val) Insert(f[p].l,val);
else Insert(f[p].r,val);
update(p);
if(Can_Rbu(p)) Rbu(p);
}
}
void del(int &p,int val)
{
if(!p) return;
f[p].sd--;
if(f[p].val==val)
{
if(f[p].cnt) f[p].cnt--;
}
else
{
if(val<f[p].val) del(f[p].l,val);
else del(f[p].r,val);
update(p);
}
if(Can_Rbu(p)) Rbu(p);
}
int Get_uprb(int p,int val)
{
if(!p) return 1;
if(f[p].val==val&&f[p].cnt) return f[f[p].l].sd+f[p].cnt+1;
else if(val<f[p].val) return Get_uprb(f[p].l,val);
else return f[f[p].l].sd+f[p].cnt+Get_uprb(f[p].r,val);
}
int Get_lwrb(int p,int val)
{
if(!p) return 0;
if(f[p].val==val&&f[p].cnt) return f[f[p].l].sd;
else if(val<f[p].val) return Get_lwrb(f[p].l,val);
else return f[f[p].l].sd+f[p].cnt+Get_lwrb(f[p].r,val);
}
int GetVal(int p,int rk)
{
if(!p) return 0;
if(rk>f[f[p].l].sd&&rk<=f[f[p].l].sd+f[p].cnt) return f[p].val;
else if(rk>f[f[p].l].sd+f[p].cnt) return GetVal(f[p].r,rk-f[f[p].l].sd-f[p].cnt);
else return GetVal(f[p].l,rk);
}
int main()
{
read(n);
for(int i=1;i<=n;i++)
{
int op;
read(op);
int x;
read(x);
switch(op)
{
case 1:
Insert(rt,x);break;
case 2:
del(rt,x);break;
case 3:
write(Get_lwrb(rt,x)+1);putchar('\n');break;
case 4:
write(GetVal(rt,x));putchar('\n');break;
case 5:
write(GetVal(rt,Get_lwrb(rt,x)));putchar('\n');break;
case 6:
write(GetVal(rt,Get_uprb(rt,x)));putchar('\n');break;
}
}
return 0;
}
原文:https://www.cnblogs.com/xiaoh105/p/12163392.html