[学习笔记]普通平衡树Splay
哈哈哈哈哈哈哈终于会打\(splay\)啦
现在我来发一下\(splay\)的讲解吧
小蒟蒻由于码风与他人不同,所以自己找了上百篇码风诡异的\(splay\)合成的,感谢\(zcysky\)的代码与我码风相近,让我看懂了
首先,\(splay\)其实就是把一棵二叉搜索树变成一棵深度不会超过\(logn\)的二叉搜索树,它在不断旋转至深度至\(logn\)
当然,那么就少不了\(rotate\)和\(splay\)操作
具体看代码好了,说的比较麻烦
#include <bits/stdc++.h>
using namespace std;
const int maxn=100000+10;
int ch[maxn][2],fa[maxn],siz[maxn],cnt[maxn],key[maxn];
int sz,rt;
inline void clear(int x){//清空结点
ch[x][0]=ch[x][1]=fa[x]=siz[x]=cnt[x]=key[x]=0;
}
inline bool get(int x){//判断是左儿子还是右儿子
return ch[fa[x]][1]==x;
}
inline void update(int x){//更新结点
if(x){
siz[x]=cnt[x];
if(ch[x][0]) siz[x]+=siz[ch[x][0]];
if(ch[x][1]) siz[x]+=siz[ch[x][1]];
}
}
inline void rotate(int x){//双旋
int y=fa[x],z=fa[y],k=get(x);
ch[y][k]=ch[x][k^1];fa[ch[y][k]]=y;
ch[x][k^1]=y;fa[y]=x;fa[x]=z;
if(z) ch[z][ch[z][1]==y]=x;
update(y);update(x);
}
inline void splay(int x){//伸展
for(int y;y=fa[x];rotate(x))
if(fa[y])
rotate((get(x)==get(y))?y:x);
rt=x;
}
inline void insert(int val){//插入
if(rt==0){sz++;ch[sz][0]=ch[sz][1]=fa[sz]=0;rt=sz;siz[sz]=cnt[sz]=1;key[sz]=val;return;}
int x=rt,y=0;
while(1){
if(val==key[x]){cnt[x]++;update(x);update(y);splay(x);return;}
y=x;x=ch[x][key[x]<val];
if(x==0){
sz++;fa[sz]=y;
ch[y][key[y]<val]=sz;
ch[sz][0]=ch[sz][1]=0;
siz[sz]=cnt[sz]=1;
key[sz]=val;
update(y);
splay(sz);
return;
}
}
}
inline int find(int val){//找x的排名
int x=rt,ans=0;
while(1){
if(val<key[x]) x=ch[x][0];
else {
ans+=(ch[x][0]?siz[ch[x][0]]:0);
if(val==key[x]){
splay(x);return ans+1;
}
ans+=cnt[x];
x=ch[x][1];
}
}
}
inline int findkth(int val){//找排名为x的数
int x=rt,k;
while(1){
if(ch[x][0]&&val<=siz[ch[x][0]])
x=ch[x][0];
else {
k=(ch[x][0]?siz[ch[x][0]]:0)+cnt[x];
if(val<=k) return key[x];
val-=k;x=ch[x][1];
}
}
}
inline int pre(){//找前驱
int x=ch[rt][0];
while(ch[x][1]) x=ch[x][1];
return x;
}
inline int nxt(){//找后继
int x=ch[rt][1];
while(ch[x][0]) x=ch[x][0];
return x;
}
inline void del(int val){//删除结点
find(val);int x=rt;
if(cnt[rt]>1){cnt[rt]--;update(rt);return;}
if(!ch[rt][0]&&!ch[rt][1]){clear(rt);rt=0;return;}
if(!ch[rt][0]){rt=ch[x][1];fa[rt]=0;clear(x);return;}
else if(!ch[rt][1]){rt=ch[x][0];fa[rt]=0;clear(x);return;}
splay(pre());
ch[rt][1]=ch[x][1];
fa[ch[x][1]]=rt;
clear(x);update(rt);
}
int main()
{
int n,opt,x;
scanf("%d",&n);
while(n--){
scanf("%d%d",&opt,&x);
switch(opt){
case 1:insert(x);break;
case 2:del(x);break;
case 3:printf("%d\n",find(x));break;
case 4:printf("%d\n",findkth(x));break;
case 5:insert(x);printf("%d\n",key[pre()]);del(x);break;
case 6:insert(x);printf("%d\n",key[nxt()]);del(x);break;
}
}
return 0;
}
[学习笔记]普通平衡树Splay的更多相关文章
- 学习笔记:平衡树-splay
嗯好的今天我们来谈谈cosplay splay是一种操作,是一种调整二叉排序树的操作,但是它并不会时时刻刻保持一个平衡,因为它会根据每一次操作把需要操作的点旋转到根节点上 所谓二叉排序树,就是满足对树 ...
- [学习笔记]平衡树(Splay)——旋转的灵魂舞蹈家
1.简介 首先要知道什么是二叉查找树. 这是一棵二叉树,每个节点最多有一个左儿子,一个右儿子. 它能支持查找功能. 具体来说,每个儿子有一个权值,保证一个节点的左儿子权值小于这个节点,右儿子权值大于这 ...
- 平衡树学习笔记(3)-------Splay
Splay 上一篇:平衡树学习笔记(2)-------Treap Splay是一个实用而且灵活性很强的平衡树 效率上也比较客观,但是一定要一次性写对 debug可能不是那么容易 Splay作为平衡树, ...
- BST,Splay平衡树学习笔记
BST,Splay平衡树学习笔记 1.二叉查找树BST BST是一种二叉树形结构,其特点就在于:每一个非叶子结点的值都大于他的左子树中的任意一个值,并都小于他的右子树中的任意一个值. 2.BST的用处 ...
- 平衡树splay学习笔记#2
讲一下另外的所有操作(指的是普通平衡树中的其他操作) 前一篇的学习笔记连接:[传送门],结尾会带上完整的代码. 操作1,pushup操作 之前学习过线段树,都知道子节点的信息需要更新到父亲节点上. 因 ...
- 平衡树学习笔记(6)-------RBT
RBT 上一篇:平衡树学习笔记(5)-------SBT RBT是...是一棵恐怖的树 有多恐怖? 平衡树中最快的♂ 不到200ms的优势,连权值线段树都无法匹敌 但是,通过大量百度,发现RBT的代码 ...
- 平衡树学习笔记(5)-------SBT
SBT 上一篇:平衡树学习笔记(4)-------替罪羊树 所谓SBT,就是Size Balanced Tree 它的速度很快,完全碾爆Treap,Splay等平衡树,而且代码简洁易懂 尤其是插入节点 ...
- 平衡树学习笔记(2)-------Treap
Treap 上一篇:平衡树学习笔记(1)-------简介 Treap是一个玄学的平衡树 为什么说它玄学呢? 还记得上一节说过每个平衡树都有自己的平衡方式吗? 没错,它平衡的方式是......rand ...
- [学习笔记]可持久化数据结构——数组、并查集、平衡树、Trie树
可持久化:支持查询历史版本和在历史版本上修改 可持久化数组 主席树做即可. [模板]可持久化数组(可持久化线段树/平衡树) 可持久化并查集 可持久化并查集 主席树做即可. 要按秩合并.(路径压缩每次建 ...
随机推荐
- JWT设计实现
一.JWT基于token的认证流程: 二.JWT SDK的选取: https://jwt.io/ 三.编写JWT Helper: 1.获取token 设置密钥,规定算法,设置过期时间,设置发行方,生命 ...
- MVVM Light 笔记
4.关于子视图, MVVMLight Using Two Views:http://www.codeproject.com/Articles/323187/MVVMLight-Using-Two-Vi ...
- java.io.IOException: java.sql.SQLException: ORA-01502: index 'BTO.PK_xxxxx' or partition of such index is in unusable state
最近由于数据库的全备出问题了,所以一直在观察. 刚好发现很多不需要的数据,就删了几百个G的数据吧. 今天突然就报这个问题. java.io.IOException: java.sql.SQLExcep ...
- 2019.01.16 bzoj3526: [Poi2014]Card(线段树)
传送门 线段树菜题. 题意:有一些卡牌,正反各有一个数,你可以任意翻转,每次操作会将两张卡牌的位置调换,你需要在每次操作后回答以现在的卡牌顺序能否通过反转形成一个单调不降的序列. 思路: 对于一个线段 ...
- pat 甲级 1086(树的遍历||建树)
思路1:可以用建树来做 由于是先序遍历,所以直接先序建树就行了. #include<iostream> #include<cstdio> #include<cstring ...
- MPA(Macro Prudential Assessment)即宏观审慎评估体系
MPA(Macro Prudential Assessment)即宏观审慎评估体系.2015年12月29日,央行发布公告称,为进一步完善宏观审慎政策框架,更加有效地防范系统性风险,发挥逆周期调节作用, ...
- adb 相关命令 以及无法adb识别设备的解决方法
[自己解决方法] 在-/.android/文件夹下面新建adb_usb.ini文件.里面写入设备的idVendor号(0x加上四位数字),然后输入 adb kill-server, 然后adb dev ...
- js setInterval详解
[自己总结]: 语法 setInterval(code,interval) ①可以有第三个参数,第三个参数作为第一个参数(函数)的参数 ②第一个参数是函数,有三种形式: 1.传函数名,不用加引号,也 ...
- DDR4控制笔记
DDR4接口 A[17:0] input 为激活命令提 供行地址,为读.写命令地址输入:提供列地址,也为模式寄存器设 置提供操作码,A[16]只用于8Gb和16Gb,A[17]只用于16Gb,另外 ...
- MySQL中@变量的妙用
背景需求:如下图所示,需要将下面为空的字段值,填充为第一行所示的值 第一次处理失败了 第二次使用成功 使用的SQL语句如下: set @tmp_var=''; select b.id,b.table_ ...