\(\texttt{0x01}\) 前言

Splay 树(伸展树)是一棵二叉搜索树,由 Daniel SleatorRobert Tarjan 于 1985 年发明。它凭借旋转可以有 $O(\log n) $ 插入,删除等的较优秀的时间复杂度。

前置芝士:普通二叉排序树

推荐博客:

\(\texttt{0x02}\) 如何构造一棵 Splay

我们定义一个结构体:

#define val(x) t[x].val
#define ls(x) t[x].ch[0]
#define rs(x) t[x].ch[1]
#define son(x,nxt) t[x].ch[nxt]
#define fa(x) t[x].fa
#define cnt(x) t[x].cnt
#define siz(x) t[x].siz
struct node{
int val,fa,ch[2],siz,cnt;
}t[N];
int root,tot;

其中构造一个新节点的函数长这样:

void newPoint(int val,int fa,int nxt){ //值为val,父节点为fa,为fa的nxt儿子
tot++;
fa(tot)=fa; cnt(tot)=siz(tot)=1; val(tot)=val;
son(fa,nxt)=tot;
}

\(\texttt{0x03}\) which / pushup / connect

which 的作用是判断 \(x\) 是其父节点的左节点(\(0\))还是右节点(\(1\)),代码很好写:

bool which(int x){
return rs(fa(x))==x;
}

pushup 的作用是维护当前节点的 \(siz\) 信息,和线段树的 pushup 性质差不多,代码:

void pushup(int x){
siz(x)=siz(ls(x))+siz(rs(x))+cnt(x);//记得加上当前节点的cnt
}

connect 的作用是把 \(x\) 变成 \(y\) 的 \(nxt\) 儿子,无需考虑覆盖的问题,代码也很简洁:

void connect(int x,int y,int nxt){
son(y,nxt)=x;
fa(x)=y;
}

\(\texttt{0x04}\) rotate

Splay 的核心操作:旋转。

放两张动图:



我们会发现:右旋时,E 节点要到 S 节点的位置上,那么 E 节点的右儿子因为它 \(\ge E\) 且 \(\le S\),所以只能放在 S 节点的左儿子,然后要改变 E 和 S 的父子关系。最后别忘了因为有旋转,所以要自下而上更新节点信息。

左旋同理。

代码:

void rotate(int x){
int y=fa(x),z=fa(y);
int fx=which(x),fy=which(y); connect(son(x,fx^1),y,fx); //如果x是左儿子,改变它右儿子的位置,反之同理
connect(y,x,fx^1); //把y接到x的缺失的那一棵子树上
connect(x,z,fy); //把x接到y的父节点上去 pushup(y); pushup(x); //别搞错顺序
}

\(\texttt{0x05}\) splay

Splay 树保证时间复杂度正确的核心操作,把 \(x\) 转到 \(y\) 的位置(\(y\) 通常为 \(root\))。

有几点结论,难证但好记:

  • 若 \(fa(x)=y\),则单旋 \(x\)。
  • 若 \(x\)、\(fa(x)\)、\(fa\left(fa(x)\right)\) 不在一条线上,则先单旋 \(fa(x)\),再单旋 \(x\)。
  • 否则旋转两次 \(x\)。
void splay(int x,int y){
y=fa(y); //避免x=y时出现的错误
while(fa(x)!=y){
if(fa(fa(x))==y) // Case 1
rotate(x);
else if(which(x)==which(fa(x))) // Case 2
rotate(fa(x)), rotate(x);
else // Case 3
rotate(x), rotate(x);
}
if(y==0){ // 如果y是根,把根变为x
root=x;
connect(x,0,1);
}
}

\(\texttt{0x06}\) insert

与普通的二叉排序树基本一致。

  • 如果树中已经有值了,则 \(cnt \gets cnt+1\)。
  • 如果找到最后都没有值,建个新节点。

记得最后要 splay 一下,把这个点转到根节点。

void insert(int val){
if(root==0){
newPoint(val,0,1);
root=tot;
return;
}
int now=root;
while(1){
siz(now)++;
if(val(now)==val){
cnt(now)++;
splay(now,root);
return;
}
int nxt=val(now)<val, son=son(now,nxt);
if(!son){
newPoint(val,now,nxt);
splay(tot,root);
return;
}
now=son;
}
}

\(\texttt{0x07}\) find

这一步操作是找到树中值为 \(val\) 的节点,并把它旋转到根节点,为 delete 操作做准备。

与普通二叉排序树也基本一致。

int find(int val){
int now=root;
while(1){
if(!now)
return 0;
if(val(now)==val){
splay(now,root);
return now;
}
int nxt=val(now)<val, son=son(now,nxt);
now=son;
}
}

\(\texttt{0x08}\) delete

目的是删除树中值为 \(val\) 的节点。

先 find 这个节点,让他转到根,然后分类讨论。

  1. 树中没有值为 \(val\) 的节点,删了个寂寞。

  2. 树中值为 \(val\) 的节点有不止一个(即 \(cnt \ge 2\)),让 \(cnt \gets cnt-1\) 即可。

  3. 这个节点没有左儿子(即根节点只有右子树),把右儿子设为根就行了。

  4. 这个节点没有右儿子(即根节点只有左子树),把左儿子设为根就行了。

  5. 这个节点(设为 \(x\))有左右儿子,把它的左子树中值最大的(设为 \(y\))splay 到根,然后现在的 Splay 树的根就是 \(y\),左子树是原来的除 \(y\) 之外的左子树,右子树是 \(x\) 和之前的右子树。把之前的右子树 connect 到根就行了。

注意:这里的删除操作都没有回收编号

void delet(int val){
int now=find(val);
if(!now) return;
if(cnt(now)>1){
cnt(now)--; siz(now)--;
return;
}
if(!ls(now) && !rs(now)){
root=0;
}
else if(!ls(now)){
root=rs(root);
fa(root)=0;
}
else if(!rs(now)){
root=ls(root);
fa(root)=0;
}
else{
int pos=ls(now);
while(rs(pos)) pos=rs(pos); splay(pos,root);
connect(rs(now),pos,1);
pushup(pos);
}
}

\(\texttt{0x09}\) rnk & find_k

rnk 是返回值为 \(val\) 的数在树中的排名,find_k 是找到树中排名为 k 的数。

与二叉排序树基本完全相同。记得最后要把节点 splay 到根。

int rnk(int val){
int now=root,s=0;
while(now){
if(val(now)==val){
splay(now,root);
return siz(ls(now))+1;
}
if(val(now)<val){
s+=siz(ls(now))+cnt(now);
now=rs(now);
}
else{
now=ls(now);
}
}
return s+1;
}
int find_k(int k){
int now=root;
while(1){
int used=siz(now)-siz(rs(now));
if(k>siz(ls(now)) && k<=used){
break;
}
if(k>=used){
k-=used;
now=rs(now);
}
else{
now=ls(now);
}
}
splay(now,root);
return val(now);
}

\(\texttt{0x0A}\) lower & upper

返回值为 \(val\) 的数的前驱和后继。

int lower(int val){
int ans=-2147483647;
int now=root;
while(now){
if(val(now)<val && val(now)>ans){
ans=val(now);
}
if(val>val(now)){
now=rs(now);
}
else{
now=ls(now);
}
}
return ans;
}
int upper(int val){
int ans=2147483647;
int now=root;
while(now){
if(val(now)>val && val(now)<ans){
ans=val(now);
}
if(val<val(now)){
now=ls(now);
}
else{
now=rs(now);
}
}
return ans;
}

\(\texttt{0x0B}\) 完整代码

#include<bits/stdc++.h>
using namespace std;
constexpr int N = 5e5+5; template <typename T> void read(T &x){x=0; T f(0); char ch=getchar(); while(ch<'0'||ch>'9'){f|=ch=='-';ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<1)+(x<<3)+(ch^48); ch=getchar();} x=f?-x:x;}
template <typename T,typename ...Arg>void read(T& x,Arg& ...arg){read(x);read(arg...);}
template <typename T> inline void write(T x){static char buf[64]; static int tot(0); if(x<0) putchar('-'),x=-x; do buf[++tot]=(x%10)+48,x/=10; while(x); do putchar(buf[tot--]); while(tot);}
template <typename T> void write(T x,char c){static char buf[64]; static int tot(0); if(x<0) putchar('-'),x=-x; do buf[++tot]=(x%10)+48,x/=10; while(x); do putchar(buf[tot--]); while(tot); putchar(c);} class Splay{
#define val(x) t[x].val
#define ls(x) t[x].ch[0]
#define rs(x) t[x].ch[1]
#define son(x,nxt) t[x].ch[nxt]
#define fa(x) t[x].fa
#define cnt(x) t[x].cnt
#define siz(x) t[x].siz
private:
struct node{
int val,fa,ch[2],siz,cnt;
}t[N];
int root,tot;
public:
bool which(int x){
return rs(fa(x))==x;
}
void pushup(int x){
siz(x)=siz(ls(x))+siz(rs(x))+cnt(x);
}
void connect(int x,int y,int nxt){
son(y,nxt)=x;
fa(x)=y;
}
void rotate(int x){
int y=fa(x),z=fa(y);
int fx=which(x),fy=which(y); connect(son(x,fx^1),y,fx);
connect(y,x,fx^1);
connect(x,z,fy); pushup(y); pushup(x);
}
void splay(int x,int y){
y=fa(y);
while(fa(x)!=y){
if(fa(fa(x))==y)
rotate(x);
else if(which(x)==which(fa(x)))
rotate(fa(x)), rotate(x);
else
rotate(x), rotate(x);
}
if(y==0){
root=x;
connect(x,0,1);
}
}
void newPoint(int val,int fa,int nxt){
tot++;
fa(tot)=fa; cnt(tot)=siz(tot)=1; val(tot)=val;
son(fa,nxt)=tot;
}
void insert(int val){
if(root==0){
newPoint(val,0,1);
root=tot;
return;
}
int now=root;
while(1){
siz(now)++;
if(val(now)==val){
cnt(now)++;
splay(now,root);
return;
}
int nxt=val(now)<val, son=son(now,nxt);
if(!son){
newPoint(val,now,nxt);
splay(tot,root);
return;
}
now=son;
}
}
int find(int val){
int now=root;
while(1){
if(!now)
return 0;
if(val(now)==val){
splay(now,root);
return now;
}
int nxt=val(now)<val, son=son(now,nxt);
now=son;
}
}
void delet(int val){
int now=find(val);
if(!now) return;
if(cnt(now)>1){
cnt(now)--; siz(now)--;
return;
}
if(!ls(now) && !rs(now)){
root=0;
}
else if(!ls(now)){
root=rs(root);
fa(root)=0;
}
else if(!rs(now)){
root=ls(root);
fa(root)=0;
}
else{
int pos=ls(now);
while(rs(pos)) pos=rs(pos); splay(pos,root);
connect(rs(now),pos,1);
pushup(pos);
}
}
int rnk(int val){
int now=root,s=0;
while(now){
if(val(now)==val){
splay(now,root);
return siz(ls(now))+1;
}
if(val(now)<val){
s+=siz(ls(now))+cnt(now);
now=rs(now);
}
else{
now=ls(now);
}
}
return s+1;
}
int find_k(int k){
int now=root;
while(1){
int used=siz(now)-siz(rs(now));
if(k>siz(ls(now)) && k<=used){
break;
}
if(k>=used){
k-=used;
now=rs(now);
}
else{
now=ls(now);
}
}
splay(now,root);
return val(now);
}
int lower(int val){
int ans=-2147483647;
int now=root;
while(now){
if(val(now)<val && val(now)>ans){
ans=val(now);
}
if(val>val(now)){
now=rs(now);
}
else{
now=ls(now);
}
}
return ans;
}
int upper(int val){
int ans=2147483647;
int now=root;
while(now){
if(val(now)>val && val(now)<ans){
ans=val(now);
}
if(val<val(now)){
now=ls(now);
}
else{
now=rs(now);
}
}
return ans;
}
}tr; int n,opt,val; int main(){
read(n);
while(n--){
read(opt,val);
int ans;
switch(opt){
case 1:{
tr.insert(val);
break;
}
case 2:{
tr.delet(val);
break;
}
case 3:{
ans=tr.rnk(val);
break;
}
case 4:{
ans=tr.find_k(val);
break;
}
case 5:{
ans=tr.lower(val);
break;
}
case 6:{
ans=tr.upper(val);
break;
}
}
if(opt>2) write(ans,'\n');
}
}

【学习笔记】Splay的更多相关文章

  1. [学习笔记] Splay Tree 从入门到放弃

    前几天由于出行计划没有更博QwQ (其实是因为调试死活调不出来了TAT我好菜啊) 伸展树 伸展树(英语:Splay Tree)是一种二叉查找树,它能在O(log n)内完成插入.查找和删除操作.它是由 ...

  2. [学习笔记]Splay

    其实就是一道题占坑啦 [NOI2005]维护数列 分析: 每次操作都要 \(Splay\) 一下 \(Insert\) 操作:重建一棵平衡树,把 \(l\) 变成根,\(l+2\) 变成右子树的根,那 ...

  3. 平衡树splay学习笔记#2

    讲一下另外的所有操作(指的是普通平衡树中的其他操作) 前一篇的学习笔记连接:[传送门],结尾会带上完整的代码. 操作1,pushup操作 之前学习过线段树,都知道子节点的信息需要更新到父亲节点上. 因 ...

  4. [学习笔记]平衡树(Splay)——旋转的灵魂舞蹈家

    1.简介 首先要知道什么是二叉查找树. 这是一棵二叉树,每个节点最多有一个左儿子,一个右儿子. 它能支持查找功能. 具体来说,每个儿子有一个权值,保证一个节点的左儿子权值小于这个节点,右儿子权值大于这 ...

  5. 平衡树学习笔记(3)-------Splay

    Splay 上一篇:平衡树学习笔记(2)-------Treap Splay是一个实用而且灵活性很强的平衡树 效率上也比较客观,但是一定要一次性写对 debug可能不是那么容易 Splay作为平衡树, ...

  6. BST,Splay平衡树学习笔记

    BST,Splay平衡树学习笔记 1.二叉查找树BST BST是一种二叉树形结构,其特点就在于:每一个非叶子结点的值都大于他的左子树中的任意一个值,并都小于他的右子树中的任意一个值. 2.BST的用处 ...

  7. 学习笔记 | CDQ分治

    目录 前言 啥是CDQ啊(它的基本思想) 例题 后记 参考博文 前言 博主太菜了 学习快一年的OI了 好像没有什么会的算法 更寒碜的是 学一样还不精一样TAT 如有什么错误请各位路过的大佬指出啊感谢! ...

  8. OI知识点|NOIP考点|省选考点|教程与学习笔记合集

    点亮技能树行动-- 本篇blog按照分类将网上写的OI知识点归纳了一下,然后会附上蒟蒻我的学习笔记或者是我认为写的不错的专题博客qwqwqwq(好吧,其实已经咕咕咕了...) 基础算法 贪心 枚举 分 ...

  9. 平衡树学习笔记(6)-------RBT

    RBT 上一篇:平衡树学习笔记(5)-------SBT RBT是...是一棵恐怖的树 有多恐怖? 平衡树中最快的♂ 不到200ms的优势,连权值线段树都无法匹敌 但是,通过大量百度,发现RBT的代码 ...

  10. 平衡树学习笔记(5)-------SBT

    SBT 上一篇:平衡树学习笔记(4)-------替罪羊树 所谓SBT,就是Size Balanced Tree 它的速度很快,完全碾爆Treap,Splay等平衡树,而且代码简洁易懂 尤其是插入节点 ...

随机推荐

  1. 大文件分片上传,后端拼接保存(前端:antd;后端:.Net 5 WebAPI)

    前言 对于普通业务场景而言,直接用 FormData() 将文件以入参的一个参数传给后端即可,但此方法有一个弊端就是,有个 30M 的上限. 对于动辄几百 M.几个 G 的文件上传需求,FormDat ...

  2. 19_Vue如何监测到对象类型数据发生改变的?

    数据更新 关于监视 我们之前讲过,我们在data当中配置的属性,最终会挂载在vue实例身上,而data这个配置项,最终也会在vue身上成为一个新的属性 == _data 当我们在页面DOM当中,去使用 ...

  3. ES6 学习笔记(四)基本类型Number

    1.数值 1.1 .JavaScript数值的特点 不区分整数值和浮点数值. 所有数值均用浮点数值表示. 采用IEEE-754标准定义的64位浮点数格式表示. 整数在实际操作时(如数组索引),则是基于 ...

  4. 谷歌浏览器xpath获取网页按钮路径

    谷歌浏览器打开要获取的页面按下F12打开开发者工具 点击最左边的元素选择器,高亮后光标移动到对应元素框(这里只选择里层的元素,如这里要选到input级别) 点击后下方HTML会高亮显示,鼠标移动上去右 ...

  5. elasticsearch聚合之bucket terms聚合

    目录 1. 背景 2. 前置条件 2.1 创建索引 2.2 准备数据 3. 各种聚合 3.1 统计人数最多的2个省 3.1.1 dsl 3.1.2 运行结果 3.2 统计人数最少的2个省 3.2.1 ...

  6. RabbitMq简单模式

    RabbitMq简单模式 定义一个生产者,负责发送消息到队列中 /** * @author zjh * 生产者发信息 */ public class Producer { /** * 队列名称 */ ...

  7. 造个Python轮子,实现根据Excel生成Model和数据导入脚本

    前言 最近遇到一个需求,有几十个Excel,每个的字段都不一样,然后都差不多是第一行是表头,后面几千上万的数据,需要把这些Excel中的数据全都加入某个已经上线的Django项目 这就需要每个Exce ...

  8. i春秋123

    打开是个普普通通的登录窗口,下尝试根据提示12341234进行输入,发现不正确...可能1234是指步骤,然后查看源码 发现了绿色的提示信息,我们就根据提示试试打开user.php 打开是白板网页,源 ...

  9. Day21:方法重写以及注意细节

    目录 方法重写 什么是方法重写? 方法重写有什么用? 方法重写的注意细节 方法重写 什么是方法重写? 方法重写指的是当子类和父类出现了一摸一样的方法声明 方法重写有什么用? 当父类中有一个方法时,子类 ...

  10. windows使用pyinstaller 打包sklearn模块出现死循环报错

    前言 解决这个让我花费了很长时间, 我这里说的死循环,不是正常通过 --hidden-import能解决的问题. 因为我也查询了很多资料 但是无一例外都失败了(能通过 –hidden-import 解 ...