参考视频:https://www.bilibili.com/video/BV1wt411u7xL?t=1142讲的特别好!
注释都在代码中了

/*Keep on going Never give up*/
#pragma GCC optimize(3,"Ofast","inline")
#include <bits/stdc++.h>
const int maxn = 2e5+10;
const int MaxN = 0x3f3f3f3f;
const int MinN = 0xc0c0c00c;
typedef long long ll;
const int mod = 100000000;
using namespace std;

struct node{   //这里记录重复结点用的是cnt这个变量1代表这个数有一个,2代表这个数有两个。。。
    int l,r;
    int val,size,cnt;
}spl[maxn];
int cnt,root;

void new_node(int &now,int &val){  //新建结点
    now=++cnt;
    spl[now].val=val;
    spl[now].size++;
    spl[now].cnt++;
}
void update(int now){   //更新size
    spl[now].size=spl[spl[now].l].size+spl[spl[now].r].size+spl[now].cnt;
}
void zig(int &now){   //右旋
    int l=spl[now].l;
    spl[now].l=spl[l].r;
    spl[l].r=now;
    now=l;
    update(spl[now].r),update(now);
}
void zag(int &now){   //左旋
    int r=spl[now].r; //把他右子树的结点暂存一下,一会需要以右孩子为根节点
    spl[now].r=spl[r].l; //把根节点右子树的左子树挂在原来根节点的上右子树上
    spl[r].l=now; //把原来的根节点挂在原来的右子树(现在的根节点)上去
    now=r;   //现在的根节点更新
    update(spl[now].l),update(now); //检查树是否需要更新
}

void splaying(int x,int &y){   //
    if(x==y) return;   //如果到了我们指定的结点,return
    //主要是通过递归进行伸展操作,
    //通过从根节点不断递归到自己想要悬上去的结点
    //不过缺点就是常数可能比较大,但是好写一些
    int &l=spl[y].l,&r=spl[y].r;
    if(x==l) zig(y);
    else if(x==r) zag(y);
    else{
        if(spl[x].val<spl[y].val){
            if(spl[x].val<spl[l].val)
                splaying(x,spl[l].l),zig(y),zig(y);     //zigzig情况
            else splaying(x,spl[l].r),zag(l),zig(y);    //zagzig情况
        }
        else{
            if(spl[x].val>spl[r].val)
                splaying(x,spl[r].r),zag(y),zag(y);     //zagzag情况
            else splaying(x,spl[r].l),zig(r),zag(y);    //zigzag情况
        }
    }
}
void del_node(int now){   //找到删除的结点并将其删除
    splaying(now,root);  //先把要删除的结点延申到根节点
    //看看次数是否重复了,如果重复了结点直接-1即可(看内存池是如何定义的)
    if(spl[now].cnt>1){
        spl[now].size--;
        spl[now].cnt--;
    }
    else if(spl[root].r){  //找一下后继,建议看一下up的视频,讲的很清晰(17min开始看)
        int temp=spl[root].r;
        while (spl[temp].l) temp=spl[temp].l;
        splaying(temp,spl[root].r);
        spl[spl[root].r].l=spl[root].l;
        root=spl[root].r;
        update(root);
    }
    else root=spl[root].l;
}
void del(int &now,int val){   //找到那个删除的结点
    if(spl[now].val==val) del_node(now);
    else if(val<spl[now].val) del(spl[now].l,val);
    else del(spl[now].r,val);
}
void insert(int &now,int val){  //插入
    if(!now) new_node(now,val),splaying(now,root);  //新建结点并插入
    else if(val<spl[now].val) insert(spl[now].l,val);
    else if(val>spl[now].val) insert(spl[now].r,val);
    else spl[now].size++,spl[now].cnt++,splaying(now,root);
}
int get_rank(int val){
    int now=root,rank=1;
    //往树的右孩子走,同时减去加上他小的个数
    //(他的左子树的数都比他小,所以他的总排名肯定要比左边的所有数都高,
    //所以先用rank加上左数大小)
    while (now){
        if(spl[now].val==val){
            rank+=spl[spl[now].l].size;
            splaying(now,root);
            break;
        }
        if(val<=spl[now].val) now=spl[now].l;
        else{
            rank+=spl[spl[now].l].size+spl[now].cnt;
            now=spl[now].r;
        }
    }
    return rank;
}
int get_num(int rank){
    int now=root;
    while (now){
        int lsize=spl[spl[now].l].size;
        if(lsize+1<=rank&&rank<=lsize+spl[now].cnt){  //如果找到了直接break
            splaying(now,root);
            break;
        }
        else if(lsize>=rank) now=spl[now].l;
        else{
            //往树的右孩子走,同时减去比他小的个数
            //(他的左子树的数都比他小,所以直接减去左子树的大小)
            rank-=lsize+spl[now].cnt;
            now=spl[now].r;
        }
    }
    return spl[now].val;
}

int main()
{
    int n;
    cin>>n;
    for(int i=0;i<n;i++){
        int ch,x;
        scanf("%d%d",&ch,&x);
        if(ch==1) insert(root,x);
        if(ch==2) del(root,x);
        if(ch==3) printf("%d\n",get_rank(x));
        if(ch==4) printf("%d\n",get_num(x));
        if(ch==5) printf("%d\n",get_num(get_rank(x)-1));
        if(ch==6) printf("%d\n",get_num(get_rank(x+1)));
    }
    return 0;
}