芝士:splay

目录

背景

因为BST本身存在一定的缺陷,

还有毒瘤出题人故意卡

导致BST极容易退化成一条单链,时间复杂度从优秀的\(O(log_n)\)到了\(O(n)\)

各种各样的巨佬就开始优化BST

所以才产生了我们所熟悉(心态爆炸)的各式各样的数据结构

巨佬Sleator和Tarjan就是其中的两位人,用旋转的方案解决了这个缺陷

主要思想

对于一个数列,我们由此生成的BST不止一种,

所以每一种的BST的高度都不一样,导致效率不一样

但是我们可以通过旋转操作使得整个树自平衡,

但是又不改变它本身BST的性质

操作

旋转

思路

我们要做的就是将当前节点的深度-1,只改变当前节点和当前节点的父亲的位置关系,但是又不改变整棵树的BST的性质,

也就是说,我们现在有一棵树,当前节点为x,现在我们想让x的深度-1

芝士:splay

成为这样一颗树

芝士:splay

一共有四种情况,笔者将其中的一种拿出来,另外3中也都是大同小异,读者可以自己手玩

代码

void rotate(int x)
{
    int y=tre[x].fa;
    int z=tre[y].fa;
    int k=tre[y].ch[1]==x;
    tre[z].ch[tre[z].ch[1]==y]=x;
    tre[x].fa=z;
    tre[y].ch[k]=tre[x].ch[k^1];
    tre[tre[x].ch[k^1]].fa=y;
    tre[x].ch[k^1]=y;
    tre[y].fa=x;
    push_up(y);
    push_up(x);
}

splay

思路

我们发现旋转操作实际上并不能使整棵树自平衡,

而实际上splay的自平衡不是来源于单纯的旋转

而是splay

我们如果我们将当前节点的祖父(父亲的父亲)节点考虑进来,

我们就会发现只有两种情况,

第一种是当前节点两次都旋转一样的方向

第二种是当前节点两次旋转的方向不一样

这里的方向定义为他是父亲节点的哪一个儿子

巨佬Sleator和Tarjan告诉我们要想自平衡,必须要遵守一个策略

如果是情况1,两次旋转都转他自己

如果是情况2,就先转他的父亲,再转他自己

通过势函数中所说的@#$!@#

可以明白splay的均摊时间复杂度为\(O(log_n)\)

在每次操作后都要用splay,因为splay有着上传的作用

代码

void splay(int x,int goal)
{
    while(tre[x].fa!=goal)
    {
        int y=tre[x].fa;
        int z=tre[y].fa;
        if(z!=goal)
        {
            if((tre[z].ch[0]==y)^(tre[y].ch[0]==x))
                rotate(x);
            else
                rotate(y);
        }
        rotate(x);
    }
    if(goal==0)
        rt=x;
    push_up(x);
}

将某个节点提到根节点

思路

直接在BST中找到这个点

再用splay转上去就好了

代码

void prepare(int x)
{
    int u=rt;
    if(!u)
        return;
    while(x!=tre[u].val&&tre[u].ch[x>tre[u].val])
    {
        if(x>tre[u].val)
        {
            u=tre[u].ch[1];
        }
        else
        {
            u=tre[u].ch[0];
        }
    }
    splay(u,0);
}

初始化

思路

因为蒟蒻笔者写的操作很多都是基于前驱与后继写的

所以最开始先插入极小值和极大值会好写很多

代码

void init()
    {
        rt=1;
        cnt=2;
        tre[0].fa=0;
        tre[0].ch[0]=tre[0].ch[1]=0;
        tre[0].tot=0;
        tre[0].val=0;
        tre[0].siz=0;

        tre[1].fa=0;
        tre[1].ch[0]=0;
        tre[1].ch[1]=2;
        tre[1].tot=1;
        tre[1].val=INT_MIN;
        tre[1].siz=2;

        tre[2].fa=1;
        tre[2].ch[0]=0;
        tre[2].ch[1]=0;
        tre[2].tot=1;
        tre[2].val=INT_MAX;
        tre[2].siz=1;
    }

前驱&后继

思路

暴力,反正时间复杂度也是\(O(log_n)\)

代码

int next(int x,int f)
{
    prepare(x);
    int u=rt;
    if(tre[u].val>x&&f)
        return u;
    if(tre[u].val<x&&!f)
        return u;
    u=tre[u].ch[f];
    while(tre[u].ch[!f])
        u=tre[u].ch[!f];
    return u;
}

插入

思路

通过旋转插入这个点的前驱和后继,

将插入的点的位置确定下来

代码

void insert(int x)
{
    int p=next(x,0);
    int s=next(x,1);
    splay(p,0);
    splay(s,p);
    if(tre[s].ch[0]==0)
    {
        tre[s].ch[0]=newnode(x,s);
        splay(cnt,0);
    }
    else
    {
        tre[tre[s].ch[0]].tot++;
        tre[tre[s].ch[0]].siz++;
        splay(tre[s].ch[0],0);  
    }
}

删除

思路

同插入一样的思路

代码

void delet(int x)
{
    int p=next(x,0);
    int s=next(x,1);
    splay(s,0);
    splay(p,s);
    if(tre[tre[p].ch[1]].tot>1)
    {
        tre[tre[p].ch[1]].tot--;
        tre[tre[p].ch[1]].siz--;
        splay(tre[p].ch[1],0);
    }
    else
    {
        tre[p].ch[1]=0;
        splay(p,0);
    }
}

排名

思路

将要询问的点转到根节点

之后查询儿子的大小就行了

代码

int rank(int x)
{
    prepare(x);
    return tre[tre[rt].ch[0]].siz;
}

第k大

思路

暴力

代码

int kth(int x)
{
    x++;
    int u=rt;
    while(1)
    {
        if(tre[tre[u].ch[0]].siz<x&&x<=tre[tre[u].ch[0]].siz+tre[u].tot)
        {
            return tre[u].val;
        }
        if(tre[tre[u].ch[0]].siz+tre[u].tot<x)
        {
            x=x-tre[tre[u].ch[0]].siz-tre[u].tot;
            u=tre[u].ch[1];
        }
        else
            u=tre[u].ch[0];
    }
}

板子

这里以洛谷P3369为例

#include<iostream>
#include<climits>
using namespace std;
struct Splay
{   
    #define MAXN 100005
    struct node
    {
        int fa;
        int ch[2];
        int tot;//相同键值的数的个数
        int val;//键值
        int siz;//个数
    }tre[MAXN];
    int rt;
    int cnt;
    #undef MAXN
    void init()
    {
        rt=1;
        cnt=2;
        tre[0].fa=0;
        tre[0].ch[0]=tre[0].ch[1]=0;
        tre[0].tot=0;
        tre[0].val=0;
        tre[0].siz=0;

        tre[1].fa=0;
        tre[1].ch[0]=0;
        tre[1].ch[1]=2;
        tre[1].tot=1;
        tre[1].val=INT_MIN;
        tre[1].siz=2;

        tre[2].fa=1;
        tre[2].ch[0]=0;
        tre[2].ch[1]=0;
        tre[2].tot=1;
        tre[2].val=INT_MAX;
        tre[2].siz=1;
    }
    int newnode(int x,int fa)
    {
        ++cnt;
        tre[cnt].fa=fa;
        tre[cnt].ch[0]=tre[cnt].ch[1]=0;
        tre[cnt].tot=1;
        tre[cnt].val=x;
        tre[cnt].siz=1;
        return cnt;
    }
    void push_up(int k)
    {
        tre[k].siz=tre[tre[k].ch[0]].siz+tre[tre[k].ch[1]].siz+tre[k].tot;
    }
    void rotate(int x)
    {
        int y=tre[x].fa;
        int z=tre[y].fa;
        int k=tre[y].ch[1]==x;
        tre[z].ch[tre[z].ch[1]==y]=x;
        tre[x].fa=z;
        tre[y].ch[k]=tre[x].ch[k^1];
        tre[tre[x].ch[k^1]].fa=y;
        tre[x].ch[k^1]=y;
        tre[y].fa=x;
        push_up(y);
        push_up(x);
    }
    void splay(int x,int goal)
    {
        while(tre[x].fa!=goal)
        {
            int y=tre[x].fa;
            int z=tre[y].fa;
            if(z!=goal)
            {
                if((tre[z].ch[0]==y)^(tre[y].ch[0]==x))
                    rotate(x);
                else
                    rotate(y);
            }
            rotate(x);
        }
        if(goal==0)
            rt=x;
        push_up(x);
    }
    void prepare(int x)
    {
        int u=rt;
        if(!u)
            return;
        while(x!=tre[u].val&&tre[u].ch[x>tre[u].val])
        {
            if(x>tre[u].val)
            {
                u=tre[u].ch[1];
            }
            else
            {
                u=tre[u].ch[0];
            }
        }
        splay(u,0);
    }
    int next(int x,int f)
    {
        prepare(x);
        int u=rt;
        if(tre[u].val>x&&f)
            return u;
        if(tre[u].val<x&&!f)
            return u;
        u=tre[u].ch[f];
        while(tre[u].ch[!f])
            u=tre[u].ch[!f];
        return u;
    }
    void insert(int x)
    {
        int p=next(x,0);
        int s=next(x,1);
        splay(p,0);
        splay(s,p);
        if(tre[s].ch[0]==0)
        {
            tre[s].ch[0]=newnode(x,s);
            splay(cnt,0);
        }
        else
        {
            tre[tre[s].ch[0]].tot++;
            tre[tre[s].ch[0]].siz++;
            splay(tre[s].ch[0],0);  
        }
    }
    void delet(int x)
    {
        int p=next(x,0);
        int s=next(x,1);
        splay(s,0);
        splay(p,s);
        if(tre[tre[p].ch[1]].tot>1)
        {
            tre[tre[p].ch[1]].tot--;
            tre[tre[p].ch[1]].siz--;
            splay(tre[p].ch[1],0);
        }
        else
        {
            tre[p].ch[1]=0;
            splay(p,0);
        }
    }
    int rank(int x)
    {
        prepare(x);
        return tre[tre[rt].ch[0]].siz;
    }
    int kth(int x)
    {
        x++;
        int u=rt;
        while(1)
        {
            if(tre[tre[u].ch[0]].siz<x&&x<=tre[tre[u].ch[0]].siz+tre[u].tot)
            {
                return tre[u].val;
            }
            if(tre[tre[u].ch[0]].siz+tre[u].tot<x)
            {
                x=x-tre[tre[u].ch[0]].siz-tre[u].tot;
                u=tre[u].ch[1];
            }
            else
                u=tre[u].ch[0];
        }
    }
}tre;
int n;
int opt,x;
int main()
{
    ios::sync_with_stdio(false);
    tre.init();
    cin>>n;
    for(int i=1;i<=n;i++)
    {
        cin>>opt>>x;
        if(opt==1)
        {
            tre.insert(x);
        }
        if(opt==2)
        {
            tre.delet(x);
        }
        if(opt==3)
        {
            cout<<tre.rank(x)<<'\n';
        }
        if(opt==4)
        {
            cout<<tre.kth(x)<<'\n'; 
        }
        if(opt==5)
        {
            cout<<tre.tre[tre.next(x,0)].val<<'\n'; 
        }
        if(opt==6)
        {
            cout<<tre.tre[tre.next(x,1)].val<<'\n';
        }
    }
    return 0;
}
上一篇:深度学习模型部署阿里云服务器


下一篇:LVM逻辑卷管理、 磁盘故障小案例