[学习笔记]普通平衡树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树
可持久化:支持查询历史版本和在历史版本上修改 可持久化数组 主席树做即可. [模板]可持久化数组(可持久化线段树/平衡树) 可持久化并查集 可持久化并查集 主席树做即可. 要按秩合并.(路径压缩每次建 ...
随机推荐
- SQL判断如果一列值为null则取另一列值代替 isnull()
[chClientCode] ,[nvcClientName] ,[chRegionCode] ,isnull(chUltimateHeadClientCode,[chClientCode]) as ...
- Python之开发自动化管理工具paramiko
一.paramiko模块使用 1)远程执行主机命令获取结果 方法一 import paramiko # 创建SSH对象 ssh = paramiko.SSHClient() # 允许连接不在know_ ...
- 3、iOS Xcode创建protocol(代理).h文件
- php代码中pre的作用??
- Codeforces 1098 简要题解
文章目录 前言 A题 B题 C题 D题 E题 传送门 前言 没错因为蒟蒻太菜了这场的最后一道题也咕掉了,只有AAA至EEE的题解233 A题 传送门 题意简述:给出一棵带点权的树,根节点深度为111, ...
- 2018.11.01 NOIP训练 木棒分组(搜索+剪枝)
传送门 测试搜索的时候状态定义错了233. 我们把木棒从大到小排序. 然后保证每一组搜到的木棒出现的长度是从大到小递减的. 直接定义现在搜的木棒从什么位置开始,当前这一组的总长度,之前几组的总长度. ...
- Win7 VS2015及MinGW环境编译矢量库agg-2.5和cairo-1.14.6
书接上文,昨天装了MinGW,主要原因之一是要用到MSYS,所以顺手把FFMPEG又编译了一遍. 回到主题,其实我是想编译矢量库,因为最近要学习一些计算几何算法,所以找个方便的2D画图库就很重要. 说 ...
- ASP.NET WebApi 基于分布式Session方式实现Token签名认证(发布版)
一.课程介绍 明人不说暗话,跟着阿笨一起学玩WebApi!开发提供数据的WebApi服务,最重要的是数据的安全性.那么对于我们来说,如何确保数据的安全将会是需要思考的问题.在ASP.NETWebSer ...
- string流
istringstream和ostringstream 从istringstream类中读取数据赋值给某个string,写入某个string到ostringstream类,头文件<sstream ...
- idea使用git提交代码到远程,这里是没有冲突的演示
首先在项目鼠标右键,找到Git,然后在Git选项里找到Add,点击: 添加到暂存区后,再次找到Git,找到Commit Directory,点击: 然后弹出这个界面: 选中你自己修改的记录,一些不必要 ...