splay入门

安利好的博客

好迷平衡树这一块....

今天先学一些其中splay的一些简单操作吧!

平衡树属于二叉查找树的一种,简单定义是:对于任意一个节点而言:左儿子内的节点都比他小,而右节点都比它大..

例如此树:

这就是一个二叉查找树...

至于splay有什么神奇的奇特之处,嗯就是旋转....我们如果要将一个点x将其与父亲y调换以下的话.就可以避免贝卡成链的情况..

共有四种情况:

    

   

这是全部的四种情况,我们发现可以找规律:

1:x变为y在z中的位置.

2.y变为x在y中相反的位置.

3.x的 与x在y中位置 相同位置的儿子与x的位置关系不变.

4:x的 与x在y中位置 相反位置的儿子变成y中与原本在x中相同的位置的儿子.(真绕....)

5:y除x的3儿子与y的相对位置不变...

用代码就是这样的:

//其实旋转操作大体分三步:
//1.将x提上去.
//2.将x的与x在y中位置不相同的儿子给y.
//3.y到x的与x在y中位置不相同的儿子的位置. 
inline void retate(int x)
{
    int y=t[x].ff;//x的父亲 
    int z=t[y].ff;//y的父亲. 
    int k=t[y].ch[1]==x;//x是y的哪一个儿子.0是左儿子,y是右儿子.
    t[z].ch[t[c].ch[1]==y]=x;//从上往下更新.原本y的位置由x顶替. 
    t[x].ff=z;//x的父亲变成z
    t[y].ch[k]=t[x].ch[k^1];//先替换x中与x在y中位置相反的儿子.防止之后被覆盖.
    t[t[x].ch[k^1]].ff=y;//更新父亲节点. 
    t[x].ch[k^1]=y;//x的与x在y中位置相反的儿子变成y. 
    t[y].ff=x; //更新父亲节点. 
} 

多画图,多动手,你会恍然大悟的...

这就是基本操作了(大雾...)

接下来思考这样的问题:

如果我们要让x一直到某一个点的儿子怎么办,一直向上旋转吗?

例如下面东西:

 

我们发现这样确实是可以的,但有一个弊端,不知道你们发现了没有?x旋转前与旋转后,始终有x,y,z,b这条链,那如果在这条链上查找的话复杂度就可能被卡成O(n^2)的.这我们不能接受.

其实再画画图就会发现,(我真的累了..),总共才两种情况...

1.x和y所属位置与y和z所属位置相同...

2.....不同

发现第一种情况与上面的样例相同,我们可以先旋转y,再旋转x,第二种情况直接旋转x即可.

inline void splay(int x,int goal)
{
    while(t[x].ff!=goal)//x一直旋转,直到是goal的儿子 
    {
        int y=t[x].ff,z=t[y].ff;//找到x的父亲,y的父亲. 
        if(y!=goal) (t[y].ch[0]==x)^(t[z].ch[0]==y):retato(y)?retato(x);//分类讨论 
        retato(x);//最后都旋转x 
    }
    if(goal==0) root=x;//更新根. 
}

好,到这里splay最核心的操作就完毕了,之后就是各个功能的应用.

首先是定义:

struct wy
{
    int val;//权值
    int ff;//父节点.
    int ch[2];//左右两个儿子.0是左儿子,1是右儿子.
    int cnt;//这个权值的节点出现的次数.
    int sum;//子节点的数量.  
}; 

 

插入操作:

当树为空时,直接新建一个节点即可.

树不空时,从根开始往下走.

inline int newpoint(int v,int fa)
{
    t[++tot].ff=fa; 
    t[tot].v=v;
    t[tot].sum=cnt=1;
    return tot;
} 
 
inline void insert(int x)
{
    int now=root;
    if(now==0) {root=newpoint(x,0);return;}//如果树为空. 
    while(1)
    {
        t[now].sum++; //依次更新子树个数. 
        if(t[now].val==x) {t[now].cnt++;splay(now,0);return;}//找到与x相同的值,累加一下数量. 
        int next=x>t[now].val;//找到下一个去的方向. 
        if(!t[now].ch[next])//如果找到最后没有相同的值,新建节点. 
        {
            int p=newpoint(x,now);
            t[now].ch[next]=p;
            splay(p,root);return;//维护形态. 
        }
        now=t[now].ch[next];//往下找. 
    }
}

查找数x的位置:

inline void find(int x)
{
    int now=root;
  if(!now) return;
  while(x!=v(now)&&t[now].ch[x>v(now)]) now=t[now].ch[x>v(now)];
  splay(now,0);
}

 

查找前驱/后继:

(这是在提前加入了INF和-INF的前提下)

inline int Next(int x,int op) //找前驱0/后继1
{
    find(x);
    int now=root;
    if(v(now)>x&&op) return now;//如果不存在x的话,find函数实质上返回了一个x的前驱或后继. 
    if(v(now)<x&&!op) return now;//所以这里就可以特判一些情况. 
    now=t[now].ch[op];//之后的情况就是存在x或find找的值与我们期望的值不对等. 
    while(t[now].ch[op^1]) now=t[now].ch[op^1];//(我们想要前驱,它找了后继,我们想要后继,它找了前驱.)
    return now;
}

 

删除操作:

删除权值为x的数:

inline void Delete(int x)
{
    int last=Next(x,0);//查找这个点的前驱 
    int next=Next(x,1);//查找这个点的后继 
    splay(last,0);splay(next,last);//将前驱搞到根.后继搞到前驱的右儿子. 
    int del=t[next].ch[0];//此时比前驱大且比后继大,只有x. 
    if(cnt(del)>=2) 
    {
        cnt(del)--;//数量大于2,减少一 
        splay(del,0);
    }
    else t[next].ch[0]=0;//否则直接搞掉. 
}

 

查找第k大;

inline int kth(int x) 
{
    int now=root;//当前根节点. 
    if(sum[x]<x) return 0;//不存在. 
    while(1)
    {
        int y=t[now].ch[0];//左儿子. 
        if(x>sum(y)+cnt(now))//左儿子里无排名第k大. 
        {
            x-=sum(y)+cnt(now);//更新排名. 
            now=t[now].ch[1];//到右儿子上 
        }
        else if(sum(y)>=x) now=y;//到左儿子 
        else return v(now);//放回这个节点. 
    }
}

 

posted @ 2020-04-23 15:06  逆天峰  阅读(211)  评论(0编辑  收藏  举报
作者:逆天峰
出处:https://www.cnblogs.com/gcfer//