哈哈哈哈哈哈哈终于会打\(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的更多相关文章

  1. 学习笔记:平衡树-splay

    嗯好的今天我们来谈谈cosplay splay是一种操作,是一种调整二叉排序树的操作,但是它并不会时时刻刻保持一个平衡,因为它会根据每一次操作把需要操作的点旋转到根节点上 所谓二叉排序树,就是满足对树 ...

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

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

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

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

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

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

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

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

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

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

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

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

  8. 平衡树学习笔记(2)-------Treap

    Treap 上一篇:平衡树学习笔记(1)-------简介 Treap是一个玄学的平衡树 为什么说它玄学呢? 还记得上一节说过每个平衡树都有自己的平衡方式吗? 没错,它平衡的方式是......rand ...

  9. [学习笔记]可持久化数据结构——数组、并查集、平衡树、Trie树

    可持久化:支持查询历史版本和在历史版本上修改 可持久化数组 主席树做即可. [模板]可持久化数组(可持久化线段树/平衡树) 可持久化并查集 可持久化并查集 主席树做即可. 要按秩合并.(路径压缩每次建 ...

随机推荐

  1. ubuntu14简介/安装/菜鸟使用手册

    Linux拥有众多的发行版,可以分为两大类商业版和开源社区免费版.商业版以Radhat为代表,开源社区版以debian为代表. 简单的比较ubuntu与centos.    Ubuntu 优点:丰富的 ...

  2. MySQL把本地数据库上传到linux

    今天是要导入数据库到linux系统 先用Navicat把sql导出,然后登陆到linux去执行 登陆mysql: mysql -uroot -p123456 创建数据库:create database ...

  3. Find the location of libmysqlclient.so.X file in Linux environments

    I'm putting together a script that has a requirement of knowing libmysqlclient.so.[15|16|18] .so fil ...

  4. day16正则表达式作业

    1.匹配一篇英文文章的标题 类似 The Voice Of China #([A-Z][a-z]*)( [A-Z][a-z]*)* 2.匹配一个网址 #(https|http|ftp):\/\/[^\ ...

  5. ATM自动取款机程序感想

    上周四的Java考试,老师并没有我们考暑假给我们布置的样卷的java程序,而是让我们做一个设计ATM的程序,然而这个对于我们来说好难,因为暑假没有学好java,首先基础知识还没有掌握,输入数据一开始都 ...

  6. MyBatis中的缓存1

    1.应用程序和数据库交互的过程是一个相对比较耗时的过程 2.缓存存在的意义:让应用程序减少对数据库的访问,提升程序运行的xiaolv 3.MyBatis中默认SqlSession缓存开启 3.1  同 ...

  7. linux挂载ntfs格式的硬盘

    发生了一件辣眼睛的操作,一个现场应用升级,由于跨度很大,不敢直接动,就把现场的数据库dump拿回来,在公司做写升级测试. 于是,联系现场的工程师把数据库dump导出来,放到网盘弄回来. ------- ...

  8. 2018.12.15 bzoj3998: [TJOI2015]弦论(后缀自动机)

    传送门 后缀自动机基础题. 求第kkk小的子串(有可能要求本质不同) 直接建出samsamsam,然后给每个状态赋值之后在上面贪心选最小的(过程可以类比主席树/平衡树的查询操作)即可. 代码: #in ...

  9. 简单实现java线程池

    使用多线程以及线程池的意义无需多说,要想掌握线程池,最好的方法还是自己手动去实现. 一.实现思路      (网络盗图) 二.实现代码 1.线程池类 package com.ty.thread; im ...

  10. vim 查找替换批量替换

    一.vi查找:    当你用vi打开一个文件后,因为文件太长,如何才能找到你所要查找的关键字呢?在vi里可没有菜单-〉查找, 不过没关系,你在命令模式下敲斜杆(/)这时在状态栏(也就是屏幕左下脚)就出 ...