平衡树之Splay

Posted sktt1faker

tags:

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

算法简介

Splay是一种平衡树,支持插入、删除、求排名、求第(k)大数、求前驱和求后继的操作,并且它还能做到一般平衡树做不到的区间操作。

定义与性质

先说二叉查找树:就是把所有数建在树上,且左边的数永远小于右边的。
对于上面说的那6个操作,其实在数据随机时二叉查找树时最强的,但是数据一条链你就Good Game了。
这种情况我们希望这棵二叉查找树的节点深度差不要太大,这就有了平衡树。
顾名思义,平衡树是平衡的二叉查找树,意思就是说1条链这种数据对于平衡树来说完全不存在,这样复杂度就有保证了。

基础操作

核心操作

这些都是很重要的操作,直接维护了Splay的平衡。

pushup

维护子树大小,很简单,不谈。好像只在旋转操作中出现。

inline void pushup(int x)
{
    siz[x]=siz[ch[x][0]]+siz[ch[x][1]]+ct[x];
}

左右旋

核心中的核心,是另外一个核心操作的基础核心操作。
这个东西可以改变两个相邻节点的父子关系,并且仍然满足平衡树的性质。给张图,就可以看明白了。
技术图片

inline void rota(int x)
{
    int y=ff[x],z=ff[y],k=(x==ch[y][1]);
    ch[z][y==ch[z][1]]=x;
    ff[x]=z;
    ch[y][k]=ch[x][k^1];
    ff[ch[x][k^1]]=y;
    ch[x][k^1]=y;
    ff[y]=x;
    pushup(x),pushup(y);
}

splay

为什么Splay要叫Splay?因为这个操作。这个操作就是直接保证了Splay的复杂度。
这个操作就是把一个节点旋转为目标节点的子节点,或者旋转到根节点。
这里情况有点小多(写成代码就短了,毕竟这个代码经过了一代又一代压行先辈的优化):
若该节点的爷爷节点为目标节点,直接把这个节点旋上去。
若爷爷节点不为目标节点,且该节点、父节点、爷爷节点三点一线,那么先旋父节点,再旋该节点。
若爷爷节点不为目标节点,且该节点、父节点、爷爷节点三点不共线,那么旋2次该节点。

inline void splay(int x,int father)
{
    while(ff[x]!=father)
    {
        int y=ff[x],z=ff[y];
        if(z!=father)
          (y==ch[z][0])^(x==ch[y][0])?rota(x):rota(y);//压行神器
        rota(x);
    }
    if(!father)  
      root=x;
}

6个操作

其实就跟二叉查找树差不多。

插入

我们给每个节点设计数器,表示这个节点代表的数有多少个。
先从根节点开始找这个数,每次都要记录目前节点及其父亲。
找到了(最后记录目前节点的变量不为(0))就说明这个数存在,存在就把该节点计数器(+1)
没找到,那么我们可以肯定刚才的父亲可以作为新数的父亲,于是新数就变成它儿子了。注意,若父亲为(0),则新数为根。
最后要把新数旋为根节点,保证随机性。

inline void Insert(int x)
{
    int now=root,father=0;
    while(now&&x!=val[now])
    {
        father=now;
        now=ch[now][x>val[now]];
    }
    if(now)
      ct[now]++;
    else 
    {
        now=++cnt;
        if(!father)
          root=now;
        else ch[father][x>val[father]]=now;
        ff[now]=father,val[now]=x,ct[now]=1,siz[now]=1;     
    }
    splay(now,0);
}

删除

先找前驱、后继的节点编号(下面会讲,可以先看下面),然后把前驱旋为根节点,后继旋为根节点的右儿子,那么此时待删除节点就被卡在后继的左儿子那儿了。
之后还是看看计数器,如果大于(1),那么就直接(-1)之后旋转该点到根节点保证随机,否则干净利落地砍掉。
为了防止找不到前驱后继,我们一开始就插入(-inf)以及(inf)

void Delete(int x)
{
    int xp=qpre(x),xs=qsuf(x);
    splay(xp,0),splay(xs,xp);
    int now=ch[xs][0];
    if(ct[now]>1)
    {
        ct[now]--;
        splay(now,0);
    }
    else ch[xs][0]=0;
}

查询第(k)大数

从根节点节点开始走,如果该节点的左子树的大小大于(k),那么继续往左子树走。
否则如果该节点左子树的大小加上该点计数器小于(k),答案就是该节点的值。
否则往左边走,同时(k)要减去该节点计数器以及该节点左子树大小。

inline int kth(int x)
{
    int now=root;
    if(siz[now]<x)
      return 0;
    while(1)
    {
        if(siz[ch[now][0]]>=x)
          now=ch[now][0];
        else if(siz[ch[now][0]]+ct[now]>=x)
          return val[now];
        else x-=siz[ch[now][0]]+ct[now],now=ch[now][1];
    }
}

查询排名

这个直接先找到该数所在节点,然后答案就是左子树大小(+1)
记得最后还要(-1),因为你插入了(-inf)

inline int qrank(int x)
{
    findx(x);
    return siz[ch[root][0]]+1;
}

前驱

找到该数节点,然后从左儿子开始一直走其右儿子,最后到的就是前驱。
为了避免找不到的问题,先插入,再删除。

inline int qpre(int x)
{
    findx(x);
    int now=ch[root][0];
    while(ch[now][1])
      now=ch[now][1];
    return now;
}

后继

同理,不多扯了。

inline int qsuf(int x)
{
    findx(x);
    int now=ch[root][1];
    while(ch[now][0])
      now=ch[now][0];
    return now;
}

它活着的意义?

我们发现,这个东西常数贼大,还长。显然,我们面对上面6种操作,用treap不行吗?
对,是。但是根据黑格尔的存在即合理理论,dzy大佬花很多时间把这个东西教给我们,教练让我花很多时间把这个东西学会,我花很多时间把这个东西学会了并写出来,评测机花很多时间去评测,那这个东西肯定在别的方面更加突出。
是的,Splay可以处理区间问题(常常是连线段树也解决不了的),而别的平衡树面对区间问题捉襟见肘。
我们处理区间问题,首先直接递归建树,建树时,节点间的大小关系不再是储存的值的关系,而是节点序号的关系。这里我们就要把节点编号设为节点在原序列中的位置。
于是我们发现,某子树的中序遍历,就为一个区间(有时可以反过来化树为区间,多用于DP)。这时每个节点就可以代表该子树的一段区间了。你还可以在节点上打lazy标记优化时间复杂度。
但是说到这里,我们发现几乎所有平衡树都有这个性质啊!那么为什么一般的平衡树很难处理区间问题?
问题在于,你很难从平衡树上找到这个区间。像treap,你怎么找?
但是Splay能找。对于区间(l~r),只需要把(l-1)旋到根节点,把(r+1)旋到根节点的右儿子,那么根节点左儿子就是要的区间。这样一个区间就被提取出来了。
事实上,一个平衡树能否能处理区间问题,就要看能否快速提取区间。于是,我们发现好像只有Splay和FHQ无旋treap(常数更小)可以做到。于是好像只有它们可以同时做到区间问题和6种操作。
事实上,还有一种数据结构也可以,那就是树状数组,不过它是主要处理区间问题,由于某些原因可以搞6种操作,在此就不提了(其实是我不会树状数组处理6种操作)。

应用

Splay可以用于LCT。尽管Splay常数比FHQ大,但是在LCT里面,不知道为什么它复杂度更小,所以用它。

代码

1、普通平衡树(6种操作)

#include<bits/stdc++.h>
using namespace std;
int n,q,root,cnt,a[100001],ff[100001],ch[100001][2],val[100001],siz[100001],ct[100001];

inline int Read()
{
    int x=0,f=1;
    char cha=getchar();
    while(!isdigit(cha))
    {
        if(cha=='-')
          f=-1;
        cha=getchar();
    }
    while(isdigit(cha))
    {
        x=(x<<3)+(x<<1)+cha-'0';
        cha=getchar();
    }
    return x*f;
}
inline void pushup(int x)
{
    siz[x]=siz[ch[x][0]]+siz[ch[x][1]]+ct[x];
}
inline void rota(int x)
{
    int y=ff[x],z=ff[y],k=(x==ch[y][1]);
    ch[z][y==ch[z][1]]=x;
    ff[x]=z;
    ch[y][k]=ch[x][k^1];
    ff[ch[x][k^1]]=y;
    ch[x][k^1]=y;
    ff[y]=x;
    pushup(x),pushup(y);
}
inline void splay(int x,int father)
{
    while(ff[x]!=father)
    {
        int y=ff[x],z=ff[y];
        if(z!=father)
          (y==ch[z][0])^(x==ch[y][0])?rota(x):rota(y);
        rota(x);
    }
    if(!father)  
      root=x;
}
inline void Insert(int x)
{
    int now=root,father=0;
    while(now&&x!=val[now])
    {
        father=now;
        now=ch[now][x>val[now]];
    }
    if(now)
      ct[now]++;
    else 
    {
        now=++cnt;
        if(!father)
          root=now;
        else ch[father][x>val[father]]=now;
        ff[now]=father,val[now]=x,ct[now]=1,siz[now]=1;     
    }
    splay(now,0);
}
inline void findx(int x)
{
    int now=root;
    if(!now)
      return;
    while(ch[now][x>val[now]]&&val[now]!=x)
      now=ch[now][x>val[now]];
    splay(now,0);
}
inline int qpre(int x)
{
    findx(x);
    int now=ch[root][0];
    while(ch[now][1])
      now=ch[now][1];
    return now;
}
inline int qsuf(int x)
{
    findx(x);
    int now=ch[root][1];
    while(ch[now][0])
      now=ch[now][0];
    return now;
}
void Delete(int x)
{
    int xp=qpre(x),xs=qsuf(x);
    splay(xp,0),splay(xs,xp);
    int now=ch[xs][0];
    if(ct[now]>1)
    {
        ct[now]--;
        splay(now,0);
    }
    else ch[xs][0]=0;
}
inline int kth(int x)
{
    int now=root;
    if(siz[now]<x)
      return 0;
    while(1)
    {
        if(siz[ch[now][0]]>=x)
          now=ch[now][0];
        else if(siz[ch[now][0]]+ct[now]>=x)
          return val[now];
        else x-=siz[ch[now][0]]+ct[now],now=ch[now][1];
    }
}
inline int qrank(int x)
{
    findx(x);
    return siz[ch[root][0]]+1;
}
int main()
{
    q=Read();
    int opt,k;
    Insert(2147483647),Insert(-2147483647);
    while(q--)
    {
        opt=Read(),k=Read();
        if(opt==1)
        {
            Insert(k);
        }
        else if(opt==2)
        {
            Delete(k);
        }
        else if(opt==3)
        {
            printf("%d
",qrank(k)-1);
        }
        else if(opt==4)
        {
            printf("%d
",kth(k+1));
        }
        else if(opt==5)
        {
            Insert(k);
            printf("%d
",val[qpre(k)]);
            Delete(k);
        }
        else 
        {
            Insert(k);
            printf("%d
",val[qsuf(k)]);
            Delete(k);
        }
    }
    return 0;
}

2、文艺平衡树(区间翻转)
还没写,先鸽着。

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

平衡树之非旋Treap

学时总结◆学时·VI◆ SPLAY伸展树

平衡二叉树之AVL树

二叉平衡树之删除节点

文艺平衡树(splay模板)

算法基础8:平衡树之红黑树