treap

Posted shzyk

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了treap相关的知识,希望对你有一定的参考价值。

zip//右旋

void zip(int &k)//右旋 
{
    int y=lc(k);
    lc(k)=rc(y);
    rc(y)=k;
    s(y)=s(k);
    upt(k);
    k=y;
}

zap//左旋

void zap(int &k)//左旋 
{
    int y=rc(k);
    rc(k)=lc(y);
    lc(y)=k;
    s(y)=s(k);
    upt(k);
    k=y;
}

insert//插入

  1.终止,添加单点信息

  2.修改size

  3.向下插入(先插后旋)

void Insert(int &k,int &key)//插入 
{
    if(!k)
    {
        k=++nt,v(k)=key,p(k)=rand();//从1开始 
        s(k)=c(k)=1,lc(k)=rc(k)=0;
        return;
    }
    s(k)++;
    if(v(k)==key) c(k)++;
    else 
        if(v(k)>key) 
        {
            Insert(lc(k),key);
            if(p(lc(k))<p(k)) zip(k); 
        } 
        else{
            Insert(rc(k),key);
            if(p(rc(k))<p(k)) zap(k); 
        }
    return;
} 
Delete//删除 
  1.查找到,删除(先旋后删)
  2.修改size
  3.向下查找
void Delete(int &k,int &key)//删除
{
    if(v(k)==key)
    {
        if(c(k)>1) c(k)--,s(k)--;
        else if(!lc(k) or !rc(k)) k=lc(k)+rc(k);
             else if(p(lc(k))<p(rc(k))) zip(k),Delete(k,key);
                     else zap(k),Delete(k,key);
        return;
    }
    s(k)--;
    if(key<v(k)) Delete(lc(k),key);
    else Delete(rc(k),key);
    return;
}

 

queryrank//查询此元素的名次(如有重复元素,返回最高名次) 
int queryrank(int &key)//查询此元素的名次(如有重复元素,返回最高名次) 
{
    int k=rt,re=0;
    while(k)
    {
        if(key==v(k)) return re+s(lc(k))+1;
        else if(key<v(k)) k=lc(k);
             else re+=s(lc(k))+c(k),k=rc(k);
    }
    return re;
}
 
querykth//查询第k小的元素
int querykth(int k)//查询第k小的元素
{
    int x=rt;
    while(x)
    {
        if(k>s(lc(x)) and k<=s(lc(x))+c(x)) return v(x);
        else if(k<=s(lc(x))) x=lc(x);
             else k-=s(lc(x))+c(x),x=rc(x); 
    }
    return v(x);
} 

 

querypre//查询元素的前驱 
int querypre(int &key)//查询元素的前驱 
{
    int x=rt,re=2e9; 
    while(x)
    {
        if(v(x)<key) re=v(x),x=rc(x);
        else x=lc(x);
    }
    return re;
}
querysur//查询元素的后驱 
int querysur(int &key)//查询元素的后驱 
{
    int x=rt,re=2e9; 
    while(x)
    {
        if(v(x)>key) re=v(x),x=lc(x);
        else x=rc(x);
    }
    return re;
} 
#include<iostream>//来个总代码(洛谷3369(https://www.luogu.org/problemnew/show/P3369))
#include<cstdio>
#include<cstdlib>
using namespace std;
void upt(int &k);
void zip(int &k);//右旋 
void zap(int &k);//左旋 
void Insert(int &k,int &key);//插入
void Delete(int &k,int &key);//删除 
int queryrank(int &key);//查询此元素的名次(如有重复元素,返回最高名次)
int querykth(int k);//查询第k小的元素
int querypre(int &key);//查询元素的前驱 
int querysur(int &key);//查询元素的后驱 
void read(int &x);
void write(int x);
struct node{
    int lc,rc,key,priority,size,cnt;
    #define lc(k) a[k].lc
    #define rc(k) a[k].rc
    #define v(k) a[k].key
    #define p(k) a[k].priority
    #define s(k) a[k].size
    #define c(k) a[k].cnt
};
node a[100050];
int nt=0,rt;//附属条件 
int main()
{
    int n;
    read(n);
    int q,s;
    for(int i=1;i<=n;i++)
    {
        read(q),read(s);
        if(q==1) Insert(rt,s);
        if(q==2) Delete(rt,s);
        if(q==3) write(queryrank(s)),putchar(
);
        if(q==4) write(querykth(s)),putchar(
);
        if(q==5) write(querypre(s)),putchar(
);
        if(q==6) write(querysur(s)),putchar(
);
    }
    return 0;
} 
void upt(int &k) {s(k)=s(lc(k))+s(rc(k))+c(k);}
void zip(int &k)//右旋 
{
    int y=lc(k);
    lc(k)=rc(y);
    rc(y)=k;
    s(y)=s(k);
    upt(k);
    k=y;
}
void zap(int &k)//左旋 
{
    int y=rc(k);
    rc(k)=lc(y);
    lc(y)=k;
    s(y)=s(k);
    upt(k);
    k=y;
}
void Insert(int &k,int &key)//插入 
{
    if(!k)
    {
        k=++nt,v(k)=key,p(k)=rand();//从1开始 
        s(k)=c(k)=1,lc(k)=rc(k)=0;
        return;
    }
    s(k)++;
    if(v(k)==key) c(k)++;
    else 
        if(v(k)>key) 
        {
            Insert(lc(k),key);
            if(p(lc(k))<p(k)) zip(k); 
        } 
        else{
            Insert(rc(k),key);
            if(p(rc(k))<p(k)) zap(k); 
        }
    return;
} 
void Delete(int &k,int &key)//删除
{
    if(v(k)==key)
    {
        if(c(k)>1) c(k)--,s(k)--;
        else if(!lc(k) or !rc(k)) k=lc(k)+rc(k);
             else if(p(lc(k))<p(rc(k))) zip(k),Delete(k,key);
                     else zap(k),Delete(k,key);
        return;
    }
    s(k)--;
    if(key<v(k)) Delete(lc(k),key);
    else Delete(rc(k),key);
    return;
}
int queryrank(int &key)//查询此元素的名次(如有重复元素,返回最高名次) 
{
    int k=rt,re=0;
    while(k)
    {
        if(key==v(k)) return re+s(lc(k))+1;
        else if(key<v(k)) k=lc(k);
             else re+=s(lc(k))+c(k),k=rc(k);
    }
    return re;
}
int querykth(int k)//查询第k小的元素
{
    int x=rt;
    while(x)
    {
        if(k>s(lc(x)) and k<=s(lc(x))+c(x)) return v(x);
        else if(k<=s(lc(x))) x=lc(x);
             else k-=s(lc(x))+c(x),x=rc(x); 
    }
    return v(x);
} 
int querypre(int &key)//查询元素的前驱 
{
    int x=rt,re=2e9; 
    while(x)
    {
        if(v(x)<key) re=v(x),x=rc(x);
        else x=lc(x);
    }
    return re;
}
int querysur(int &key)//查询元素的后驱 
{
    int x=rt,re=2e9; 
    while(x)
    {
        if(v(x)>key) re=v(x),x=lc(x);
        else x=rc(x);
    }
    return re;
}
void read(int &x)
{
    x=0;
    int i=1;
    char c=getchar();
    while(c<0 or c>9)
    {
        if(c==-) i=-1;
        c=getchar();
    }
    while(c>=0 and c<=9)
    {
        x=x*10+c-0;
        c=getchar();
    }
    x*=i;
}
void write(int x)
{
    if(x<0) putchar(-),x*=-1;
    if(x>9) write(x/10);
    putchar(x%10+0);
}

 





以上是关于treap的主要内容,如果未能解决你的问题,请参考以下文章

Treap

bzoj3224普通平衡树——treap

非旋Treap

Treap树模板hdu-4585

算法学习Fhq-Treap(无旋Treap)

无旋treap hfq-treap