引入

在一条链中,二叉查找树的时间复杂度就会退化成 \(O(n)\),这时我们就需要平衡树来解决这个问题。

\(Splay\)(伸展树)是平衡树的一种,它的每一步插入、查找和删除的平摊时间都是 \(O(\log n)\),出于对编程复杂度和算法性能的考虑,这是 OI 中常用的算法。

性质

\(Splay\) 本质上还是对二叉查找树的优化。所以它也具备二叉查找树的性质,即左子树任意节点的值 \(<\) 根节点的值 \(<\) 右子树任意节点的值

操作

数组含义

root tot fa[i] ch[i][0] ch[i][1] val[i] size[i] cnt[i]
根节点编号 节点数量 父节点编号 左儿子编号 右儿子编号 节点权值 子树大小 节点权值出现次数

基本操作

maintain(x):维护子树大小

void Splay::maintain(int x)
{
size[x] = size[ch[x][0]] + size[ch[x][1]] + cnt[x];
return ;
}

get(x):查询该节点是其父亲节点的左子树还是右子树

bool Splay::get(int x)
{
if( x == ch[fa[x]][1] )
return 1;
return 0;
}

clear(x):清理该节点

void Splay::clear(int x)
{
ch[x][0] = ch[x][1] = fa[x] = val[x] = size[x] = cnt[x] = 0;
return ;
}

旋转

旋转操作实际上是让某一个节点上移一个位置。

旋转操作需要保证,二叉查找树的性质不会改变,节点维护的信息依然正确,\(root\) 必须指向旋转后的根节点。

若节点 x 是其父亲的左节点

由于 \(x\) 的右儿子的权值大于 \(x\) 的权值,且 \(x\) 及其子树都属于 \(y\) 的左子树(即 \(x\) 的右子树实际上小于 \(y\) 的权值),所以我们将 \(x\) 的右子树改为 \(y\) 的左子树。

  1. 将 \(x\) 的右儿子变成 \(y\) 的左儿子,如果 \(x\) 有右儿子的话就让它的父亲变成 \(y\)。

    ch[y][0] = ch[x][1]; fa[ch[x][1]] = y;

由于 \(y\) 及其子树的权值都大于 \(x\) 的权值,所以我们让 \(y\) 成为 \(x\) 的右儿子。

  1. 使 \(y\) 成为 \(x\) 的右儿子,\(x\) 变为 \(y\) 的父亲。

    ch[x][chk^1] = y; fa[y] = x;

  2. 如果 \(x\) 此时不是根节点,那么 \(x\) 将继承原先 \(y\) 作为 \(z\) 的儿子的位置(\(x\) 取代 \(y\) 成为 \(z\) 的左儿子或右儿子)。

    fa[x] = z; if(z) ch[z][y == ch[z][1]] = x;

由此我们得到了节点 \(x\) 上升一个位置的树,显然,这棵树仍然满足二叉搜索树的性质。

实现

void Splay::rotate(int x)
{
int y = fa[x],z = fa[y],chk = get(x); ch[y][chk] = ch[x][chk ^ 1]; if( ch[x][chk ^ 1] )
fa[ch[x][chk ^ 1]] = y;
ch[x][chk ^ 1] = y; fa[y] = x;
fa[x] = z; if(z)
ch[z][y == ch[z][1]] = x; maintain(y);
// 要先维护 y 的子树大小
// 因为现在 y 是 x 的儿子
maintain(x);// 别忘了维护子树大小 return ;
}

代码中采用异或来实现左右不同旋转情况,当然我们可以写两个函数分别来实现左旋和右旋。

伸展

伸展操作是在保持伸展树性质的前提下,将节点 \(x\) 转移到根节点。在这个转移过程中,我们分为三种情况

首先我们设节点 \(x\) 的父节点为节点 \(y\),若节点 \(y\) 有父节点,其父节点为 \(z\)。

第一种情况:\(y\) 是根节点

  • 若 \(x\) 是 \(y\) 的左儿子,我们进行一次右旋操作

  • 若 \(x\) 是 \(y\) 的右儿子,我们进行一次左旋操作

第二种情况:\(y\) 不是根节点,且 \(x\) 和 \(y\) 同为左儿子或右儿子

  • 若 \(x\) 和 \(y\) 同时是各自父节点的左儿子,则进行两次右旋操作
  • 若 \(x\) 和 \(y\) 同时是各自父节点的右儿子,则进行两次左旋操作

第三种情况:\(y\) 不是根节点,且 \(x\) 和 \(y\) 一个为左儿子一个为右儿子

  • 若 \(x\) 是 \(y\) 的左儿子,\(y\) 是 \(z\) 的右儿子,则进行一次右旋 - 左旋操作
  • 若 \(x\) 是 \(y\) 的右儿子,\(y\) 是 \(z\) 的左儿子,则进行一次左旋 - 右旋操作

实现

void Splay::splay(int x)
{
for(int i = fa[x];i = fa[x],i; rotate(x))
if( fa[i] )
{
if( get(x) == get(i) )
rotate(i);
else
rotate(x);
} root = x; return ;
}

插入

  1. 如果树为空,则直接插入根节点

  2. 如果找到了一个节点权值与插入权值相等,则增大该节点并维护信息,再进行 Splay 操作

  3. 否则接着往下找,要是找到空节点就直接插入

实现

void Splay::insert(int v)
{
if( root == 0 )
{
tot ++;
val[tot] = v;
cnt[tot] ++;
root = tot;
maintain(root); return ;
} int cur = root,x = 0;
while(1)
{
if( val[cur] == v )
{
cnt[cur] ++;
maintain(cur);
maintain(x);
splay(cur); break;
}
x = cur;
cur = ch[cur][val[cur] < v]; if( cur == 0 )
{
tot ++;
val[tot] = v;
cnt[tot] ++;
fa[tot] = x;
ch[x][val[x] < v] = tot; maintain(tot);
maintain(x);
splay(tot); break;
}
}
}

寻找数 \(x\) 的排名(比它小的数的个数值 + 1)

  1. 若 \(x\) 小于当前节点权值,则向左子树查找

  2. 若 \(x\) 大于当前节点权值,则答案加上左子树大小 size[i] 和当前节点权值出现次数 cnt[i]

  3. 若找到与 \(x\) 相等的节点,则返回当前答案 \(+ 1\)

实现

int Splay::find_rank(int v)
{
int ans = 0,cur = root;
while(1)
{
if( v < val[cur] )
cur = ch[cur][0];
else
{
ans += size[ch[cur][0]];
if( v == val[cur] )
{
splay(cur);
return ans + 1;
}
ans += cnt[cur];
cur = ch[cur][1];
}
}
}

寻找排名为 \(x\) 的数的值

\(v\) 表示剩余排名,在初始排名的条件下不断减少。

  1. 若左子树不为空且剩余排名 \(v\) 小于等于左子树大小 \(size\)(即 \(x\) 在左子树),向左子树查找

  2. 否则减去左子树大小和根的出现次数作为剩余排名 \(v\)。若 \(v\leq 0\),则返回根节点,否则向右子树查找。

实现

int Splay::find_num(int v)
{
int cur = root;
while(1)
{
if( ch[cur][0] != 0 && v <= size[ch[cur][0]] )
cur = ch[cur][0];
else
{
v -= cnt[cur] + size[ch[cur][0]];//.//
if( v <= 0 )
{
splay(cur);
return val[cur];
}
cur = ch[cur][1];
}
}
}

查询前驱(小于 \(x\) 的最大的数)

先插入节点 \(x\),这样 \(x\) 就处在了根节点的位置。

此时 \(x\) 的左子树都小于 \(x\),寻找 \(x\) 的左子树的最右边节点即小于 \(x\) 的最大的数。

实现

int Splay::pre()
{
int cur = ch[root][0];
if( cur == 0 )
return cur;
while( ch[cur][1] )
cur = ch[cur][1]; splay(cur);
return cur;
}

查询后继(大于 \(x\) 的最小的数)

基本思想与查询前驱相同。

先插入节点 \(x\),这样 \(x\) 就处在了根节点的位置。

此时 \(x\) 的右子树都大于 \(x\),寻找 \(x\) 的右子树的最左边节点即大于 \(x\) 的最小的数。

实现

int Splay::next()
{
int cur = ch[root][1];
if( cur == 0 )
return cur;
while( ch[cur][0] )
cur = ch[cur][0]; splay(cur);
return cur;
}

合并

对于合并两棵树,其中一棵树的值都小于另一棵树的值。

我们可以找到较小一棵树的最大值 \(x\),将其旋转到根节点。

再把较大一棵树作为 \(x\) 的右子树插入。

删除

  1. 首先将 \(x\) 转移到根节点

  2. 若 \(x\) 值不只一个,即 \(cnt[x] > 1\),则直接减一退出即可。

  3. 否则将它的左右两棵子树合并

实现

void Splay::del(int v)
{
find_rank(v);///// if( cnt[root] > 1 )
{
cnt[root] --;
maintain(root); return ;
} if( ch[root][0] == 0 && ch[root][1] == 0 )
{
clear(root);
root = 0;
return ;
} if( ch[root][0] == 0 )
{
int cur = root;
root = ch[root][1];
fa[root] = 0;
clear(cur); return ;
} if( ch[root][1] == 0 )
{
int cur = root;
root = ch[root][0];
fa[root] = 0;
clear(cur); return ;
} int cur = root;
int x = pre();
fa[ch[cur][1]] = x;
ch[x][1] = ch[cur][1];
clear(cur); maintain(root); return ;
}

模板题

Luogu P3369 【模板】普通平衡树

完整代码
#include<bits/stdc++.h>

using namespace std;

const int MAXN = 114514; 

int n;

int root;
// 根节点
int ch[MAXN][2],fa[MAXN];
// 子节点( 0 左 1 右 ) 父节点
int val[MAXN];
// 权值
int size[MAXN];
// 子树大小
int cnt[MAXN];
// 这个权值出现的次数
int tot;
// 节点个数 struct Splay{
void maintain(int x);
// 维护子树大小
bool get(int x);
// 查找这个节点是父亲的左子树还是右子树
void clear(int x);
// 销毁这个节点
void rotate(int x);
// 旋转
void splay(int x);
// 伸展操作
void insert(int v);
// 插入数 v
int find_rank(int v);
// 查询数 v 的排名
int find_num(int v);
// 查询排名为 v 的数
int pre();
// 查询根节点的前驱
int next();
// 查询根节点的后继
void del(int v);
// 删除 v
}tree; void Splay::maintain(int x)
{
size[x] = size[ch[x][0]] + size[ch[x][1]] + cnt[x];
return ;
} bool Splay::get(int x)
{
if( x == ch[fa[x]][1] )
return 1;
return 0;
} void Splay::clear(int x)
{
ch[x][0] = 0;
ch[x][1] = 0;
fa[x] = 0;
val[x] = 0;
size[x] = 0;
cnt[x] = 0; return ;
} void Splay::rotate(int x)
{
int y = fa[x],z = fa[y],chk = get(x); ch[y][chk] = ch[x][chk ^ 1]; if( ch[x][chk ^ 1] )
fa[ch[x][chk ^ 1]] = y;
ch[x][chk ^ 1] = y; fa[y] = x;
fa[x] = z; if(z)
ch[z][y == ch[z][1]] = x; maintain(y);
maintain(x); return ;
} void Splay::splay(int x)
{
for(int i = fa[x];i = fa[x],i; rotate(x))
if( fa[i] )
{
if( get(x) == get(i) )
rotate(i);
else
rotate(x);
} root = x; return ;
} void Splay::insert(int v)
{
if( root == 0 )
{
tot ++;
val[tot] = v;
cnt[tot] ++;
root = tot;
maintain(root); return ;
} int cur = root,x = 0;
while(1)
{
if( val[cur] == v )
{
cnt[cur] ++;
maintain(cur);
maintain(x);
splay(cur); break;
}
x = cur;
cur = ch[cur][val[cur] < v]; if( cur == 0 )
{
tot ++;
val[tot] = v;
cnt[tot] ++;
fa[tot] = x;
ch[x][val[x] < v] = tot; maintain(tot);
maintain(x);
splay(tot); break;
}
}
} int Splay::find_rank(int v)
{
int ans = 0,cur = root;
while(1)
{
if( v < val[cur] )
cur = ch[cur][0];
else
{
ans += size[ch[cur][0]];
if( v == val[cur] )
{
splay(cur);
return ans + 1;
}
ans += cnt[cur];
cur = ch[cur][1];
}
}
} int Splay::find_num(int v)
{
int cur = root;
while(1)
{
if( ch[cur][0] != 0 && v <= size[ch[cur][0]] )
cur = ch[cur][0];
else
{
v -= cnt[cur] + size[ch[cur][0]];//.//
if( v <= 0 )
{
splay(cur);
return val[cur];
}
cur = ch[cur][1];
}
}
} int Splay::pre()
{
int cur = ch[root][0];
if( cur == 0 )
return cur;
while( ch[cur][1] )
cur = ch[cur][1]; splay(cur);
return cur;
} int Splay::next()
{
int cur = ch[root][1];
if( cur == 0 )
return cur;
while( ch[cur][0] )
cur = ch[cur][0]; splay(cur);
return cur;
} void Splay::del(int v)
{
find_rank(v);///// if( cnt[root] > 1 )
{
cnt[root] --;
maintain(root); return ;
} if( ch[root][0] == 0 && ch[root][1] == 0 )
{
clear(root);
root = 0;
return ;
} if( ch[root][0] == 0 )
{
int cur = root;
root = ch[root][1];
fa[root] = 0;
clear(cur); return ;
} if( ch[root][1] == 0 )
{
int cur = root;
root = ch[root][0];
fa[root] = 0;
clear(cur); return ;
} int cur = root;
int x = pre();
fa[ch[cur][1]] = x;
ch[x][1] = ch[cur][1];
clear(cur); maintain(root); return ;
} int main()
{
scanf("%d",&n);
for(int i = 1,opt,x;i <= n; i++)
{
scanf("%d%d",&opt,&x);
if( opt == 1 )
tree.insert(x);
else if( opt == 2 )
tree.del(x);
else if( opt == 3 )///
printf("%d\n",tree.find_rank(x));
else if( opt == 4 )////
printf("%d\n",tree.find_num(x));
else if( opt == 5 )
{
tree.insert(x);
printf("%d\n",val[tree.pre()]);
tree.del(x);
}
else
{
tree.insert(x);
printf("%d\n",val[tree.next()]);
tree.del(x);
}
}
return 0;
}

伸展树(Splay)详解的更多相关文章

  1. 树-伸展树(Splay Tree)

    伸展树概念 伸展树(Splay Tree)是一种二叉排序树,它能在O(log n)内完成插入.查找和删除操作.它由Daniel Sleator和Robert Tarjan创造. (01) 伸展树属于二 ...

  2. 纸上谈兵: 伸展树 (splay tree)[转]

    作者:Vamei 出处:http://www.cnblogs.com/vamei 欢迎转载,也请保留这段声明.谢谢!  我们讨论过,树的搜索效率与树的深度有关.二叉搜索树的深度可能为n,这种情况下,每 ...

  3. K:伸展树(splay tree)

      伸展树(Splay Tree),也叫分裂树,是一种二叉排序树,它能在O(lgN)内完成插入.查找和删除操作.在伸展树上的一般操作都基于伸展操作:假设想要对一个二叉查找树执行一系列的查找操作,为了使 ...

  4. 高级搜索树-伸展树(Splay Tree)

    目录 局部性 双层伸展 查找操作 插入操作 删除操作 性能分析 完整源码 与AVL树一样,伸展树(Splay Tree)也是平衡二叉搜索树的一致,伸展树无需时刻都严格保持整棵树的平衡,也不需要对基本的 ...

  5. 『动善时』JMeter基础 — 32、JMeter察看结果树组件详解

    目录 1.察看结果树介绍 2.察看结果树界面详解 3.察看结果树的其他功能 (1)将数据写入文件中 (2)Search功能 (3)Scroll automatically选项 4.总结 1.察看结果树 ...

  6. 伸展树(Splay tree)的基本操作与应用

    伸展树的基本操作与应用 [伸展树的基本操作] 伸展树是二叉查找树的一种改进,与二叉查找树一样,伸展树也具有有序性.即伸展树中的每一个节点 x 都满足:该节点左子树中的每一个元素都小于 x,而其右子树中 ...

  7. 【BBST 之伸展树 (Splay Tree)】

    最近“hiho一下”出了平衡树专题,这周的Splay一直出现RE,应该删除操作指针没处理好,还没找出原因. 不过其他操作运行正常,尝试用它写了一道之前用set做的平衡树的题http://codefor ...

  8. [Splay伸展树]splay树入门级教程

    首先声明,本教程的对象是完全没有接触过splay的OIer,大牛请右上角.. 首先引入一下splay的概念,他的中文名是伸展树,意思差不多就是可以随意翻转的二叉树 PS:百度百科中伸展树读作:BoGa ...

  9. 伸展树Splay【非指针版】

    ·伸展树有以下基本操作(基于一道强大模板题:codevs维护队列): a[]读入的数组;id[]表示当前数组中的元素在树中节点的临时标号;fa[]当前节点的父节点的编号;c[][]类似于Trie,就是 ...

  10. splay详解(一)

    前言 Spaly是基于二叉查找树实现的, 什么是二叉查找树呢?就是一棵树呗:joy: ,但是这棵树满足性质—一个节点的左孩子一定比它小,右孩子一定比它大 比如说 这就是一棵最基本二叉查找树 对于每次插 ...

随机推荐

  1. Java中方法的定义和使用

    方法的定义和使用 注意事项: 1.方法与方法之间是 平级关系 不可以嵌套定义 2.方法的位置 可以在类{}中任意位置 3.方法定义之后 之后被调用 才能被执行 4.return 关键字的作用  返回关 ...

  2. P1980 [NOIP2013 普及组] 计数问题

    题目链接:https://www.luogu.com.cn/problem/P1980 术语 以下的英文术语均可以翻译为数字. digit: 一个数字字符,十进制就是 0-9 之间的一个字符: num ...

  3. 在Winform分页控件中集成保存用户列表显示字段及宽度调整设置

    在Winform的分页控件里面,我们提供了很多丰富的功能,如常规分页,中文转义.导出Excel.导出PDF等,基于DevExpress的样式的分页控件,我们在其上面做了不少封装,以便更好的使用,其中就 ...

  4. 2021-09-26:搜索旋转排序数组。整数数组 nums 按升序排列,数组中的值 互不相同 。在传递给函数之前,nums 在预先未知的某个下标 k(0 <= k < nums.length)上进行了

    2021-09-26:搜索旋转排序数组.整数数组 nums 按升序排列,数组中的值 互不相同 .在传递给函数之前,nums 在预先未知的某个下标 k(0 <= k < nums.lengt ...

  5. Java基础--数据结构

    数据结构 Java工具包提供了强大的数据结构.在Java中的数据结构主要包括以下几种接口和类: 枚举(Enumeration).位集合(BitSet).向量(Vector).栈(Stack).字典(D ...

  6. protoBuf 实现客户端与服务端

    转载请注明出处: 1.定义消息格式 在 src/main/proto 目录下创建 person.proto 文件,并定义消息格式,例如: syntax = "proto3"; pa ...

  7. 自己动手写Docker学习笔记

    零.前言 本文为<自己动手写 Docker>的学习,对于各位学习 docker 的同学非常友好,非常建议买一本来学习. 书中有摘录书中的一些知识点,不过限于篇幅,没有全部摘录 (主要也是懒 ...

  8. 关于SpringBoot AutoConfiguration

    (1)如何导入的自动配置类 首先我们得从@SpringBootApplication注解入手. @SpringBootApplication public class SpringBootDemoAp ...

  9. LLM探索:GPT概念与几个常用参数 Top-k, Top-p, Temperature

    前言 上一篇文章介绍了几个开源LLM的环境搭建和本地部署,在使用ChatGPT接口或者自己本地部署的LLM大模型的时候,经常会遇到这几个参数,本文简单介绍一下~ temperature top_p t ...

  10. LINQ检索使用

    我看网上对LINQ的讲解 自己整合了一下 是语言集成查询(Language Integrated Query)是一组用于C#和Visual Basic语言的扩展.能够允许编写C#或VB代码以查询数据相 ...