【模板】平衡树——Treap和Splay
二叉搜索树($BST$):一棵带权二叉树,满足左子树的权值均小于根节点的权值,右子树的权值均大于根节点的权值。且左右子树也分别是二叉搜索树。(如下)
$BST$的作用:维护一个有序数列,支持插入$x$,删除$x$,查询排名为$x$的数,查询$x$的排名,求$x$的前驱后继等操作。
时间复杂度:$O(操作数\times 树深度)$。
也就是插入一个有序序列时复杂度稳定在$O(N^2)$……
平衡树:深度稳定在$O(log{节点数})$的$BST$。
使深度稳定的几种方法:增加一个破坏单调性的第二权值($Treap$),每插入一个数进行旋转保持平衡($Splay$),维护每个子树的$size$并使左右子树的$size$保持平衡($SBT$)等。
本文主要给出$Treap$和$Splay$的实现方法。
$Treap$:顾名思义,该数据结构是$Tree$与$Heap$的结合体。
思想:在第一关键字满足$BST$性质的同时,为每个节点随机生成一个第二关键字,并通过旋转使得第二关键字满足堆性质。
旋转:(网上讲的很清楚了w)分为左右旋两种,如图(图源网络):
例如:(图源网络,图中点内是第一关键字【满足$BST$】,点外是随机生成的第二关键字【满足堆】)
优点:常数小,实现简单。
缺点:应用范围较小,略有$0.001$%运气因素(能随机出来$10^5$个递增的数就可以去买彩票了w)
代码:
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<ctime> using namespace std;
#define MAXN 100005
#define MAXM 500005
#define INF 0x7fffffff
#define ll long long struct Treap{
int l,r; //左儿子、右儿子
int num,rnd; //该节点的第一关键字(权值)、该节点的第二关键字
int cnt,siz; //该节点权值的出现次数、以该节点为根的子树的大小
}tr[MAXN];
int tot,root; //当前节点数、当前根节点 inline int read(){
int x=,f=;
char c=getchar();
for(;!isdigit(c);c=getchar())
if(c=='-')
f=-;
for(;isdigit(c);c=getchar())
x=x*+c-'';
return x*f;
} inline void update(int k){
tr[k].siz=tr[k].cnt;
tr[k].siz+=tr[tr[k].l].siz;
tr[k].siz+=tr[tr[k].r].siz;
return;
}
inline void zig(int &k){ //将以k为根的子树左旋(看图)
int tp=tr[k].r;
tr[k].r=tr[tp].l; //将k的右儿子置为k的右儿子的左儿子
tr[tp].l=k; //将k的右儿子的左儿子置为k
tr[tp].siz=tr[k].siz; //右儿子成为新的根,size等于k的size
update(k); //更新k的size
k=tp; //以k为根的子树变为以k的右儿子为根的子树,换根
return;
}
inline void zag(int &k){ //将以k为根的子树右旋(同上)
int tp=tr[k].l;
tr[k].l=tr[tp].r;
tr[tp].r=k;
tr[tp].siz=tr[k].siz;
update(k);
k=tp;return;
}
inline void ins(int x,int &k){ //插入数x
if(k==){ //当前节点为空则在此处新建节点
k=++tot;
tr[k].cnt=tr[k].siz=;
tr[k].rnd=rand();
tr[k].num=x;
return;
}
tr[k].siz++; //插入的节点在该子树内,size+1
if(x==tr[k].num) tr[k].cnt++; //如果该数已经出现过则不用新建节点,将该节点的cnt+1即可
else if(x<tr[k].num){
ins(x,tr[k].l); //x小于当前节点的关键字则插入当前节点的左子树
if(tr[tr[k].l].rnd<tr[k].rnd) zag(k);
//如果左儿子的第二关键字不满足小根堆性质就把左儿子转上来,容易证明此时一定满足堆性质
}
else{
ins(x,tr[k].r); //x大于当前节点的关键字则插入当前节点的右子树
if(tr[tr[k].r].rnd<tr[k].rnd) zig(k); //同上
}
return;
} inline void del(int x,int &k){ //删除数x
if(k==) return; //如果x没出现则返回
if(x==tr[k].num){
if(tr[k].cnt>) tr[k].cnt--,tr[k].siz--;
//如果该节点出现次数>=1则不用移除节点,出现次数-1即可
else if(tr[k].l*tr[k].r==)
k=tr[k].l+tr[k].r;
//如果该节点的儿子数<=1则可以直接删除,即拿它的儿子代替它
else if(tr[tr[k].l].rnd<tr[tr[k].r].rnd) zag(k),del(x,k);
else zig(k),del(x,k);
//否则将该节点旋转到可以直接删除的位置再删除
return;
}
tr[k].siz--; //删除的节点在该子树内,size-1
if(x<tr[k].num) del(x,tr[k].l); //x在当前节点的左子树
else del(x,tr[k].r); //x在当前节点的右子树
return;
} inline int qrnk(int x,int k){ //查询x数的排名(相当于查询有多少个数小于x)
if(k==) return ;
if(x==tr[k].num) return tr[tr[k].l].siz+;
//找到了x,此时小于x的数的个数等于左子树的大小,排名需要+1
else if(x<tr[k].num) return qrnk(x,tr[k].l);
//x在当前节点的左子树中,直接递归左子树
else return qrnk(x,tr[k].r)+tr[tr[k].l].siz+tr[k].cnt;
//x在当前节点的右子树中,此时该节点及其左子树的权值均小于x,需要将这部分size加入答案
} inline int qnum(int x,int k){ //查询排名为x的数
if(k==) return ;
if(tr[tr[k].l].siz<x && x<=tr[tr[k].l].siz+tr[k].cnt) return tr[k].num;
//此时的排名正好确定在当前节点(大于等于当前节点的权值第一次出现的位置,小于等于该权值最后一次出现的位置),返回该节点的权值(第一关键字)即可
else if(tr[tr[k].l].siz>=x) return qnum(x,tr[k].l);
// 排名为x的数在当前节点的左子树中,直接递归
else return qnum(x-(tr[tr[k].l].siz+tr[k].cnt),tr[k].r);
//排名为x的数在当前节点的右子树中,此时该节点及其左子树不影响右子树中数的排名,需要减去这部分size
} inline int qpre(int x,int k){ //查询x数的前驱(最大的小于x的数)
if(k==) return -INF;
if(x<=tr[k].num) return qpre(x,tr[k].l);
//x在当前节点的左子树中,此时该节点不影响答案,递归左子树
else return max(qpre(x,tr[k].r),tr[k].num);
//x在当前节点的右子树中,此时该节点的权值小于等于x,又因为该节点的权值大于该节点左子树中的所有权值,将答案与k取max即可
} inline int qnxt(int x,int k){ //查询x数的后继(最小的大于x的数),基本同上
if(k==) return INF;
if(x>=tr[k].num) return qnxt(x,tr[k].r);
else return min(qnxt(x,tr[k].l),tr[k].num);
} int main(){
srand(time());
int T=read();
while(T--){
int op=read(),x=read();
switch(op){
case :ins(x,root);break;
case :del(x,root);break;
case :printf("%d\n",qrnk(x,root));break;
case :printf("%d\n",qnum(x,root));break;
case :printf("%d\n",qpre(x,root));break;
case :printf("%d\n",qnxt(x,root));break;
}
}return ;
}
$Splay$:又名旋转树,该数据结构通过巧妙的双旋&单旋($splay$)使树保持平衡。
基本思想:每次插入/查找一个节点时便将其旋转到根,在旋转过程中使树“看起来”逐渐平衡。
旋转:同上,双旋时注意若三点一线则需要转中间节点不然会失衡。(例如图中$1,2,4$节点需要先转$2$)
优点:使用范围很广,可以维护各种奇怪的区间操作。
缺点:实现复杂,常数较大,时间复杂度大概在$O(N\times log^2 N)$左右。严格证明我也不会
例题:同上。
代码:(某同学没有要求就不加注释了,需要注释可以@我w)
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio> using namespace std;
#define MAXN 100005
#define MAXM 500005
#define INF 0x7fffffff
#define ll long long struct node{
int v,f,siz,cnt,ch[];
}tr[MAXN];
int rt,tot; inline int read(){
int x=,f=;
char c=getchar();
for(;!isdigit(c);c=getchar())
if(c=='-')
f=-;
for(;isdigit(c);c=getchar())
x=x*+c-'';
return x*f;
} inline bool getf(int k){return tr[tr[k].f].ch[]==k;}
inline void update(int k){
tr[k].siz=tr[k].cnt;
tr[k].siz+=tr[tr[k].ch[]].siz;
tr[k].siz+=tr[tr[k].ch[]].siz;
return;
}
inline void clear(int k){
tr[k].v=tr[k].f=;
tr[k].ch[]=tr[k].ch[]=;
tr[k].siz=tr[k].cnt=;
return;
}
inline void rotate(int k){
int f1=tr[k].f,f2=tr[f1].f;bool d=getf(k);
tr[f1].ch[d]=tr[k].ch[d^];tr[tr[k].ch[d^]].f=f1;
tr[k].ch[d^]=f1;tr[f1].f=k;tr[k].f=f2;
if(f2) tr[f2].ch[tr[f2].ch[]==f1]=k;
update(f1);update(k);return;
}
inline void splay(int k){
for(int fa;fa=tr[k].f;rotate(k))
if(tr[fa].f)
rotate(getf(k)==getf(fa)?fa:k);
rt=k;return;
}
inline int qrnk(int x){
int now=rt,ans=;
while(){
if(x==tr[now].v){
ans+=tr[tr[now].ch[]].siz+;
splay(now);return ans;
}
else if(x<tr[now].v) now=tr[now].ch[];
else ans+=tr[tr[now].ch[]].siz+tr[now].cnt,now=tr[now].ch[];
}
}
inline int qnum(int x){
int now=rt;
while(){
if(tr[tr[now].ch[]].siz<x && tr[tr[now].ch[]].siz+tr[now].cnt>=x)
return tr[now].v;
else if(tr[tr[now].ch[]].siz>=x) now=tr[now].ch[];
else x-=tr[tr[now].ch[]].siz+tr[now].cnt,now=tr[now].ch[];
}
}
inline int qpre(){
int now=tr[rt].ch[];
while(tr[now].ch[]) now=tr[now].ch[];
return now;
}
inline int qnxt(){
int now=tr[rt].ch[];
while(tr[now].ch[]) now=tr[now].ch[];
return now;
}
inline void ins(int x){
if(!rt){
tr[++tot].v=x,tr[tot].f=;
tr[tot].ch[]=tr[tot].ch[]=;
tr[tot].siz=tr[tot].cnt=;
rt=tot;return;
}
int now=rt,fa=;
while(){
if(x==tr[now].v){
tr[now].cnt++;
update(now);update(fa);
splay(now);break;
}
fa=now;now=tr[now].ch[x>tr[now].v];
if(!now){
tr[++tot].v=x,tr[tot].f=fa;
tr[tot].ch[]=tr[tot].ch[]=;
tr[tot].siz=tr[tot].cnt=;
tr[fa].ch[x>tr[fa].v]=tot;
update(fa);splay(tot);
break;
}
}
return;
}
inline void del(int x){
qrnk(x);
if(tr[rt].cnt>) tr[rt].cnt--,update(rt);
else if(!tr[rt].ch[] && !tr[rt].ch[]) clear(x),rt=;
else if(!tr[rt].ch[]){
int tp=rt;rt=tr[rt].ch[];
tr[rt].f=;clear(tp);
}
else if(!tr[rt].ch[]){
int tp=rt;rt=tr[rt].ch[];
tr[rt].f=;clear(tp);
}
else{
int tp=rt;splay(qpre());
tr[rt].ch[]=tr[tp].ch[];
tr[tr[tp].ch[]].f=rt;
update(rt);clear(tp);
}
return;
} int main(){
int T=read();
while(T--){
int opt=read(),x=read();
switch(opt){
case :ins(x);break;
case :del(x);break;
case :printf("%d\n",qrnk(x));break;
case :printf("%d\n",qnum(x));break;
case :ins(x);printf("%d\n",tr[qpre()].v);del(x);break;
case :ins(x);printf("%d\n",tr[qnxt()].v);del(x);break;
}
}
return ;
}
【模板】平衡树——Treap和Splay的更多相关文章
- 算法模板——平衡树Treap 2
实现功能:同平衡树Treap 1(BZOJ3224 / tyvj1728) 这次的模板有了不少的改进,显然更加美观了,几乎每个部分都有了不少简化,尤其是删除部分,这个参照了hzwer神犇的写法,在此鸣 ...
- bzoj3223 文艺平衡树 (treap or splay分裂+合并)
3223: Tyvj 1729 文艺平衡树 Time Limit: 10 Sec Memory Limit: 128 MB Submit: 3313 Solved: 1883 [Submit][S ...
- [日常摸鱼]bzoj3224普通平衡树-Treap、Splay、01Trie、替罪羊树…
http://www.lydsy.com/JudgeOnline/problem.php?id=3224 经典的平衡树模板题-各种平衡树好像都可以(黄学长之前好像还用vector卡过了这题) 所以这篇 ...
- 算法模板——平衡树Treap
实现功能如下——1. 插入x数2. 删除x数(若有多个相同的数,因只删除一个)3. 查询x数的排名(若有多个相同的数,因输出最小的排名)4. 查询排名为x的数5. 求x的前驱(前驱定义为小于x,且最大 ...
- luoguP3369[模板]普通平衡树(Treap/SBT) 题解
链接一下题目:luoguP3369[模板]普通平衡树(Treap/SBT) 平衡树解析 #include<iostream> #include<cstdlib> #includ ...
- 启发式合并&线段树合并/分裂&treap合并&splay合并
启发式合并 有\(n\)个集合,每次让你合并两个集合,或询问一个集合中是否存在某个元素. 我们可以用平衡树/set维护集合. 对于合并两个\(A,B\),如果\(|A|<|B|\),那么 ...
- 普通平衡树Treap(含旋转)学习笔记
浅谈普通平衡树Treap 平衡树,Treap=Tree+heap这是一个很形象的东西 我们要维护一棵树,它满足堆的性质和二叉查找树的性质(BST),这样的二叉树我们叫做平衡树 并且平衡树它的结构是接近 ...
- 2021.12.06 平衡树——Treap
2021.12.06 平衡树--Treap https://www.luogu.com.cn/blog/HOJQVFNA/qian-xi-treap-ping-heng-shu 1.二叉搜索树 1.1 ...
- hiho #1325 : 平衡树·Treap
#1325 : 平衡树·Treap 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Ho:小Hi,我发现我们以前讲过的两个数据结构特别相似. 小Hi:你说的是哪两个啊? ...
随机推荐
- vue项目刷新当前页面
场景: 有时候我们在vue项目页面做了一些操作,需要刷新一下页面. 解决的办法及遇到的问题: this.$router.go(0).这种方法虽然代码很少,只有一行,但是体验很差.页面会一瞬间的白屏,体 ...
- 配置composer代理
composer config -g repo.packagist composer https://packagist.phpcomposer.com
- CSS3学习笔记(3)—左右飞入的文字
前几天看到一个企业招聘的动画觉得很炫,里面有个企业介绍的文字是用飞入的效果做出来的,今天尝试了写了一下,感觉还不错~\(≧▽≦)/~啦啦啦 下面来看我做的动态效果: 其实上面的效果很简单的,我的截图软 ...
- druid.io 海量实时OLAP数据仓库 (翻译+总结) (1)——分析框架如hive或者redshift(MPPDB)、ES等
介绍 我是NDPmedia公司的大数据OLAP的资深高级工程师, 专注于OLAP领域, 现将一个成熟的可靠的高性能的海量实时OLAP数据仓库介绍给大家: druid.io NDPmedia在2014年 ...
- js 改变对象的引用地址
在业务处理中我们经常会碰到列表中有编辑和新增按钮,为了能够提高代码的公用性,我们经常会使用同一组件处理. 这样会出现一个问题就是编辑的时候直接把对象传过去,直接赋值,引用地址是同一个,所以不管修改了那 ...
- 基于候选区域的R-CNN系列网络简介
使用候选区域方法(region proposal method)创建目标检测的感兴趣区域(ROI).在选择性搜索(selective search,SS)中,首先将每个像素作为一组.然后,计算每一组的 ...
- Batch Normalization层
Batch Normalization的加速作用体现在两个方面:一是归一化了每层和每维度的scale,所以可以整体使用一个较高的学习率,而不必像以前那样迁就小scale的维度:二是归一化后使得更多的权 ...
- IO多路复用模型之epoll实现机制
设想一下如下场景:有100万个客户端同时与一个服务器进程保持着TCP连接.而每一时刻,通常只有几百上千个TCP连接是活跃的(事实上大部分场景都是这种情况).如何实现这样的高并发? 在select/po ...
- docker --help 详解
[root@c1 _src]# dockerd --help Usage: dockerd [OPTIONS] A self-sufficient runtime for containers. Op ...
- sublimelinter-php 错误代码提示
先安装 SublimeLinter 如同其他插件一样使用 Package Control 来安装. 按下 Ctrl+Shift+p 进入 Command Palette 输入install进入 Pac ...