bzoj1503: [NOI2004]郁闷的出纳员 fhqtreap版

时间:2021-08-02 17:49:03

这道题写法和之前差不多 但是fhqtreap在加点的时候为了同时维护大根堆以及二叉排序树的性质所以插入时也要注意分裂

fhqteap需要判断指针是否为空 不然就会re 这个我调了很久

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int M=;
int read(){
int ans=,f=,c=getchar();
while(c<''||c>''){if(c=='-') f=-; c=getchar();}
while(c>=''&&c<=''){ans=ans*+(c-''); c=getchar();}
return ans*f;
}
int n,m,sum,mn,leave,delta;
struct node{
node *l,*r;
int sz,v,rnd;
void init(int w){sz=; v=w; rnd=rand();}
void up(){
sz=;
if(l) sz+=l->sz;
if(r) sz+=r->sz;
}
void split(node*&lw,node*&rw,int w){
if(!this){lw=; rw=; return ;}
if(w<=v){
l->split(lw,l,w);
rw=this;
}
else{
r->split(r,rw,w);
lw=this;
}
up();
}
int find(int rank){
if(!this) return ;
int ls=l?l->sz:;
if(ls+==rank) return v;
else if(ls>=rank) return l->find(rank);
else return r->find(rank-ls-);
} }tr[M],*rt;
node* merge(node*a,node*b){
if(!a) return b;
if(!b) return a;
if(a->rnd>b->rnd){
a->r=merge(a->r,b);
a->up();
return a;
}{
b->l=merge(a,b->l);
b->up();
return b;
}
}
int main()
{ char ch[];
int w;
n=read(); mn=read();
while(n--){
scanf("%s",ch); w=read();
if(ch[]=='I'&&w>=mn){
tr[++sum].init(w-delta);
node *p1,*p2;
rt->split(p1,p2,w-delta);
rt=merge(merge(p1,tr+sum),p2);
}
if(ch[]=='A') delta+=w;
if(ch[]=='S'){
node *p;
delta-=w; rt->split(p,rt,mn-delta);
if(p) leave+=p->sz;
}
if(ch[]=='F'){
if(!rt||w>rt->sz) { printf("-1\n"); continue;}
printf("%d\n",rt->find(rt->sz-w+)+delta);
}
}
printf("%d\n",leave);
return ;
}