传送门:

Treap 版本:

//OJ 1999
//by Cydiater
//2016.8.30
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <string>
#include <queue>
#include <map>
#include <ctime>
#include <cmath>
#include <algorithm>
#include <string>
#include <iomanip>
using namespace std;
#define ll long long
#define up(i,j,n)    for(int i=j;i<=n;i++)
#define down(i,j,n)    for(int i=j;i>=n;i--)
const int MAXN=1e5+5;
const int oo=0x3f3f3f3f;
inline ll read(){
    char ch=getchar();ll x=0,f=1;
    while(ch>'9'||ch<'0'){if(ch=='-')f=-1;ch=getchar();}
    while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();}
    return x*f;
}
int N,op,num,root=0,tol=0,ans=0;
struct node{
    int leftt,rightt,v,cnt,rnd,siz;
}t[MAXN];
namespace solution{
    void updata(int k){
        t[k].siz=t[t[k].leftt].siz+t[t[k].rightt].siz+t[k].cnt;
    }
    void lefturn(int &k){
        int tt=t[k].rightt;t[k].rightt=t[tt].leftt;t[tt].leftt=k;
        t[tt].siz=t[k].siz;updata(k);k=tt;
    }
    void righturn(int &k){
        int tt=t[k].leftt;t[k].leftt=t[tt].rightt;t[tt].rightt=k;
        t[tt].siz=t[k].siz;updata(k);k=tt;
    }
    void insert(int &k,int x){
        if(k==0){
            tol++;k=tol;t[k].siz=t[k].cnt=1;
            t[k].v=x;t[k].rnd=rand();
            return;
        }
        t[k].siz++;
        if(t[k].v==x)        t[k].cnt++;
        else if(x>t[k].v){
            insert(t[k].rightt,x);
            if(t[t[k].rightt].rnd<t[k].rnd)lefturn(k);
        }else{
            insert(t[k].leftt,x);
            if(t[t[k].leftt].rnd<t[k].rnd)righturn(k);
        }
    }
    void del(int &k,int x){
        if(k==0)    return;
        if(t[k].v==x){
            if(t[k].cnt>1){t[k].cnt--;t[k].siz--;return;}
            if(t[k].leftt*t[k].rightt==0)k=t[k].leftt+t[k].rightt;
            else if(t[t[k].leftt].rnd<t[t[k].rightt].rnd){
                righturn(k);del(k,x);
            }else{
                lefturn(k);del(k,x);
            }
        }else if(x>t[k].v){
            t[k].siz--;
            del(t[k].rightt,x);
        }else{
            t[k].siz--;
            del(t[k].leftt,x);
        }
    }
    int query_rank(int k,int x){
        if(k==0)return 0;
        if(t[k].v==x)        return t[t[k].leftt].siz+1;
        else if(x>t[k].v)    return t[t[k].leftt].siz+t[k].cnt+query_rank(t[k].rightt,x);
        else                 return query_rank(t[k].leftt,x);
    }
    int query_num(int k,int x){
        if(k==0)                                return 0;
        if(x<=t[t[k].leftt].siz)                return query_num(t[k].leftt,x);
        else if(x>t[t[k].leftt].siz+t[k].cnt)    return query_num(t[k].rightt,x-(t[t[k].leftt].siz+t[k].cnt));
        else                                     return t[k].v;
    }
    void query_pre(int k,int x){
        if(k==0)        return;
        if(t[k].v<x){
            ans=t[k].v;
            query_pre(t[k].rightt,x);
        }else query_pre(t[k].leftt,x);
    }
    void query_nxt(int k,int x){
        if(k==0)        return;
        if(t[k].v>x){
            ans=t[k].v;
            query_nxt(t[k].leftt,x);
        }else query_nxt(t[k].rightt,x);
    }
    void slove(){
        N=read();
        up(i,1,N){
            op=read();num=read();
            if(op==1)insert(root,num);
            if(op==2)del(root,num);
            if(op==3)printf("%d\n",query_rank(root,num));
            if(op==4)printf("%d\n",query_num(root,num));
            if(op==5){
                query_pre(root,num);
                printf("%d\n",ans);
            }
            if(op==6){
                query_nxt(root,num);
                printf("%d\n",ans);
            }
        }
    }
}
int main(){
    //freopen("input.in","r",stdin);
    using namespace solution;
    slove();
    return 0;
}

Splay 版本:

//BZOJ 3224
//by Cydiater
//2016.9.3
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <string>
#include <algorithm>
#include <queue>
#include <map>
#include <ctime>
#include <cmath>
#include <iomanip>
using namespace std;
#define ll long long
#define up(i,j,n)        for(int i=j;i<=n;i++)
#define down(i,j,n)        for(int i=j;i>=n;i--)
const int MAXN=1e6+5;
inline int read(){
    char ch=getchar();int x=0,f=1;
    while(ch>'9'||ch<'0'){if(ch=='-')f=-1;ch=getchar();}
    while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();}
    return x*f;
}
int root=0,tol=0,N,op,num;
struct SplayTree{
    int son[2],siz,cnt,fa,v;
}t[MAXN];
namespace solution{
    inline int get(int x){return x==t[t[x].fa].son[1];}//get left or right son
    inline void clear(int x){
        t[x].son[0]=t[x].son[1]=t[x].cnt=t[x].v=t[x].fa=t[x].siz=0;
    }//reset node after deleted
    inline void updata(int x){//updata node after rotated
        if(x){
            t[x].siz=t[x].cnt;
            if(t[x].son[0])t[x].siz+=t[t[x].son[0]].siz;
            if(t[x].son[1])t[x].siz+=t[t[x].son[1]].siz;
        }
    }
    inline void rotate(int x){//rotate now node to root
        int old=t[x].fa,oldf=t[old].fa,which=get(x);
        t[old].son[which]=t[x].son[which^1];t[t[old].son[which]].fa=old;
        t[old].fa=x;t[x].son[which^1]=old;
        t[x].fa=oldf;
        if(oldf)t[oldf].son[t[oldf].son[1]==old]=x;
        updata(old);updata(x);
    }
    inline void splay(int x){
        for(int fa;(fa=t[x].fa);rotate(x))if(t[fa].fa)
        rotate((get(x)==get(fa)?fa:x));root=x;
    }
    inline void insert(int v){
        if(root==0){
            root=++tol;
            t[root].son[0]=t[root].son[1]=t[root].fa=0;
            t[root].v=v;t[root].cnt=t[root].siz=1;
            return;
        }
        int now=root,fa=0;
        while(1){
            if(t[now].v==v){
                t[now].cnt++;updata(now);updata(fa);
                splay(now);break;
            }
            fa=now;now=t[now].son[t[now].v<v];
            if(now==0){
                now=++tol;
                t[now].son[0]=t[now].son[1]=0;t[now].v=v;
                t[now].siz=t[now].cnt=1;t[now].fa=fa;
                t[fa].son[t[fa].v<v]=tol;
                updata(fa);
                splay(now);
                break;
            }
        }
    }
    inline int find(int v){
        int ans=0,now=root;
        while(1){
            if(v<t[now].v)now=t[now].son[0];
            else{
                ans+=(t[now].son[0]?t[t[now].son[0]].siz:0);
                if(v==t[now].v){splay(now);return ans+1;}
                ans+=t[now].cnt;
                now=t[now].son[1];
            }
        }
    }
    inline int get_rank(int x){
        int now=root;
        while(1){
            if(t[now].son[0]&&x<=t[t[now].son[0]].siz)now=t[now].son[0];
            else{
                int tmp=(t[now].son[0]?t[t[now].son[0]].siz:0)+t[now].cnt;
                if(x<=tmp)    return t[now].v;
                x-=tmp;now=t[now].son[1];
            }
        }
    }
    inline int pre(){
        int now=t[root].son[0];
        while(t[now].son[1])now=t[now].son[1];
        return now;
    }
    inline int nxt(){
        int now=t[root].son[1];
        while(t[now].son[0])now=t[now].son[0];
        return now;
    }
    inline void del(int x){
        int whatever=find(x);
        if(t[root].cnt>1){
            t[root].cnt--;
            t[root].siz--;
            return;
        }
        if(t[root].son[0]+t[root].son[1]==0){
            clear(root);root=0;
            return;
        }
        if(!t[root].son[0]){
            int oldroot=root;root=t[root].son[1];t[root].fa=0;
            clear(oldroot);return;
        }else if(!t[root].son[1]){
            int oldroot=root;root=t[root].son[0];t[root].fa=0;
            clear(oldroot);return;
        }
        int leftbig=pre(),oldroot=root;
        splay(leftbig);
        t[t[oldroot].son[1]].fa=root;
        t[root].son[1]=t[oldroot].son[1];
        clear(oldroot);
        updata(root);
        return;
    }
    void debug(int now){
        printf("now=%d t[now].son[0]=%d t[now].son[1]=%d t[now].siz=%d t[now].cnt=%d t[now].fa=%d t[now].v=%d\n",now,t[now].son[0],t[now].son[1],t[now].siz,t[now].cnt,t[now].fa,t[now].v);
        if(t[now].son[0])debug(t[now].son[0]);
        if(t[now].son[1])debug(t[now].son[1]);
    }
}
int main(){
    //freopen("input.in","r",stdin);
    using namespace solution;
    N=read();
    while(N--){
        op=read();num=read();
        if(op==1)insert(num);
        if(op==2)del(num);
        if(op==3)printf("%d\n",find(num));
        if(op==4)printf("%d\n",get_rank(num));
        if(op==5){
            insert(num);printf("%d\n",t[pre()].v);del(num);
        }
        if(op==6){
            insert(num);printf("%d\n",t[nxt()].v);del(num);
        }
        //debug(root);
        //puts("");
    }
    return 0;
}

关于Splay写的很好的一个博客:

传送门

BZOJ3224 普通平衡树的更多相关文章

  1. [BZOJ3224]普通平衡树(旋转treap,STL-vector)

    3224: Tyvj 1728 普通平衡树 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 20328  Solved: 8979[Submit][St ...

  2. [bzoj3224]普通平衡树/3223文艺平衡树

    这是一道很普通的题.. 最近花了很多时间来想要去干什么,感觉自己还是太拿衣服 做这道题是因为偶尔看到了lavender的blog和她的bzoj早期AC记录,就被题目深深地吸引到了,原因有二: 自己sp ...

  3. BZOJ3224普通平衡树【Splay】

    3224: Tyvj 1728 普通平衡树 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 11751  Solved: 5013 Descriptio ...

  4. [TYVJ1728/BZOJ3224]普通平衡树-替罪羊树

    Problem 普通平衡树 Solution 本题是裸的二叉平衡树.有很多种方法可以实现.这里打的是替罪羊树模板. 此题极其恶心. 前驱后继模块需要利用到rank模块来换一种思路求. 很多细节的地方容 ...

  5. [转载]无旋treap:从好奇到入门(例题:bzoj3224 普通平衡树)

    转载自ZZH大佬,原文:http://www.cnblogs.com/LadyLex/p/7182491.html 今天我们来学习一种新的数据结构:无旋treap.它和splay一样支持区间操作,和t ...

  6. [您有新的未分配科技点]无旋treap:从好奇到入门(例题:bzoj3224 普通平衡树)

    今天我们来学习一种新的数据结构:无旋treap.它和splay一样支持区间操作,和treap一样简单易懂,同时还支持可持久化. 无旋treap的节点定义和treap一样,都要同时满足树性质和堆性质,我 ...

  7. bzoj3224 普通平衡树(c++vector)

    Tyvj 1728 普通平衡树 2014年8月23日6,4365 Description 您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作:1. 插入x数2. 删除x数(若有 ...

  8. BZOJ3224普通平衡树——非旋转treap

    题目: 此为平衡树系列第一道:普通平衡树您需要写一种数据结构,来维护一些数,其中需要提供以下操作:1. 插入x数2. 删除x数(若有多个相同的数,因只删除一个)3. 查询x数的排名(若有多个相同的数, ...

  9. BZOJ3224普通平衡树——旋转treap

    题目: 此为平衡树系列第一道:普通平衡树您需要写一种数据结构,来维护一些数,其中需要提供以下操作:1. 插入x数2. 删除x数(若有多个相同的数,因只删除一个)3. 查询x数的排名(若有多个相同的数, ...

  10. BZOJ3224普通平衡树

    洛谷题面链接 很早就过了,太久没打了,原本是在noip前用来练emacs的手感的. noip炸了,就滚回来更博客了(安排的计数任务刷不动,学不会容斥,打发时间...) 众所周知,splay是个好算法, ...

随机推荐

  1. Android浮动小球与开机自启动

    看着手机上的360浮动小球,不评价其具体的功能与实用性,至少在UI设计与交互方面是个不小的创新. 如图片左上角所示,球中还会显示当前手机的运行状况,向下拉动还会有弹射来达到加速.清理等目的. 那好,先 ...

  2. SqlServer导入数据到MySql

    1.下载MySql ODBC Driver并进行安装.例如我下载的这个安装包是mysql-connector-odbc-5.1.6-win32.msi. 2.装完后,添加odbc数据源: 3.在sql ...

  3. HTML5之CSS3 3D transform 剖析式学习之一

    最近坐地铁发现“亚洲动物基金”在地铁上做了很多公益广告,比较吸引人的是一个月熊的广告.做的很可爱.回去就搜了一下,发现这个网站是HTML5做的,非常炫. 所以想学习一下,方法就是传统的学习办法,模仿. ...

  4. 后缀树(BZOJ3238TLE)

    #include<cstdio> #include<cstring> #define LL long long ],stt[]; LL ans; ,sidcnt,lastcre ...

  5. C#套接字和windowsAPI套接字

    C#服务器端 第一步:用指定的端口号和服务器的ip建立一个EndPoint对像:第二步:建立一个Socket对像:第三步:用socket对像的Bind()方法绑定EndPoint:第四步:用socke ...

  6. Go语言interface详解

    interface Go语言里面设计最精妙的应该算interface,它让面向对象,内容组织实现非常的方便,当你看完这一章,你就会被interface的巧妙设计所折服. 什么是interface 简单 ...

  7. zabbix 微信报警

    http://blog.csdn.net/wh211212/article/details/52735236 Zabbix可以通过多种方式把告警信息发送到指定人,常用的有邮件,短信报警方式,但是越来越 ...

  8. Swift开发小技巧--扫描二维码,二维码的描边与锁定,设置扫描范围,二维码的生成(高清,无码,你懂得!)

    二维码的扫描,二维码的锁定与描边,二维码的扫描范围,二维码的生成(高清,无码,你懂得!),识别相册中的二维码 扫描二维码用到的三个重要对象的关系,如图: 1.懒加载各种类 // MARK: - 懒加载 ...

  9. 【NOIP 2004】虫食算

    因为一天机房都是断网状态,校内的小V评测这道题总显示Unaccept,所以下午放学后就和xiaoyimi晚上回家自习,来做一做这道题. 搜索+剪枝优化: 一开始我是顺着低位向高位填数,这么暴力在Vij ...

  10. nginx正向代理,反向代理,透明代理(总结)

    1正向代理 正向代理,也就是传说中的代理,他的工作原理就像一个跳板, 简单的说, 我是一个用户,我访问不了某网站,但是我能访问一个代理服务器 这个代理服务器呢,他能访问那个我不能访问的网站 于是我先连 ...