Tyvj 1728 普通平衡树

Description
您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作:
1. 插入x数
2. 删除x数(若有多个相同的数,因只删除一个)
3. 查询x数的排名(若有多个相同的数,因输出最小的排名)
4. 查询排名为x的数
5. 求x的前驱(前驱定义为小于x,且最大的数)
6. 求x的后继(后继定义为大于x,且最小的数)

Input
第一行为n,表示操作的个数,下面n行每行有两个数opt和x,opt表示操作的序号(1<=opt<=6)

Output
对于操作3,4,5,6每行输出一个数,表示对应答案

Sample Input
10
1 106465
4 1
1 317721
1 460929
1 644985
1 84185
1 89851
6 81968
1 492737
5 493598

Sample Output
106465
84185
492737

HINT
1.n的数据范围:n<=100000
2.每个数的数据范围:[-2e9,2e9]

题解
随便用什么数据结构(splay,treap……)
反正是很裸的数据结构题
我用的是范浩强treap

代码

#include <cstdio>
#include <cstdlib>
#include <algorithm>
using namespace std;

typedef pair<int,int> pii;

const int sor='y'+'c'+'z'+' '+'i'+'s'+' '+'o'+'u'+'r'+' '+'s'+'u'+'n';//随机数种子,orz ycz大神

struct treap
{ 
    int tot,root,son[3000009][2],size[3000009],val[3000009];

    int updata(int a)
    {
        return size[a]=size[son[a][0]]+size[son[a][1]]+1;
    }

    int random(int limit)
    {
        return rand()%limit+1;
    }

    int merge(int a,int b) //合并两棵树
    {
        if ((a==0)||(b==0))
        {
            return a+b;
        }
        if (random(size[a]+size[b])<=size[a])
        {
            son[a][1]=merge(son[a][1],b);
            updata(a);
            return a;
        }
        else
        {
            son[b][0]=merge(a,son[b][0]);
            updata(b);
            return b;
        }
    }

    pii split(int a,int k)//拆分一棵树
    {
        if (size[a]==k)
        {
            return make_pair(a,0);
        }
        if (k==0)
        {
            return make_pair(0,a);
        }
        pii t;
        if (size[son[a][0]]>=k)
        {
            t=split(son[a][0],k);
            son[a][0]=t.second;
            updata(a);
            return make_pair(t.first,a);
        }
        else
        {
            t=split(son[a][1],k-size[son[a][0]]-1);
            son[a][1]=t.first;
            updata(a);
            return make_pair(a,t.second);
        }
    }

    int rank(int v)
    {
        int i=root,ans=0;
        while (i!=0)
        {
            if (val[i]>=v)
            {
                i=son[i][0];
            }
            else
            {
                ans+=size[son[i][0]]+1;
                i=son[i][1];
            }
        }
        return ans+1;
    }

    int id(int v,int k)
    {
        if (k==0)
        {
            return 0;
        }
        if (size[son[v][0]]>=k)
        {
            return id(son[v][0],k);
        }
        if (size[son[v][0]]+1==k)
        {
            return v;
        }
        return id(son[v][1],k-size[son[v][0]]-1);
    }

    int getval(int v)
    {
        return val[id(root,v)];
    }

    int pre(int v)
    {
        return val[id(root,rank(v)-1)];
    }

    int next(int v)
    {
        return val[id(root,rank(v+1))];
    }

    int insert(int v)
    {
        tot++;
        son[tot][0]=son[tot][1]=0;
        val[tot]=v;
        size[tot]=1;
        pii t=split(root,rank(v)-1);
        root=merge(merge(t.first,tot),t.second);
        return root;
    }

    int adelete(int v)
    {
        int x;
        x=rank(v);
        pii s,t;
        s=split(root,x-1);
        t=split(s.second,1);
        root=merge(s.first,t.second);
        return root;
    }
};

treap tree;
int n,opt,x;

int main()
{
    srand(sor);
    scanf("%d",&n);
    while (n--)
    {
        scanf("%d%d",&opt,&x);
        if (opt==1)
        {
            tree.insert(x);
        }
        if (opt==2)
        {
            tree.adelete(x);
        }
        if (opt==3)
        {
            printf("%d\n",tree.rank(x));
        }
        if (opt==4)
        {
            printf("%d\n",tree.getval(x));
        }
        if (opt==5)
        {
            printf("%d\n",tree.pre(x));
        }
        if (opt==6)
        {
            printf("%d\n",tree.next(x));
        }
    }
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值