二叉搜索树($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)

例题:bzoj3224普通平衡树

代码:

#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的更多相关文章

  1. 算法模板——平衡树Treap 2

    实现功能:同平衡树Treap 1(BZOJ3224 / tyvj1728) 这次的模板有了不少的改进,显然更加美观了,几乎每个部分都有了不少简化,尤其是删除部分,这个参照了hzwer神犇的写法,在此鸣 ...

  2. bzoj3223 文艺平衡树 (treap or splay分裂+合并)

    3223: Tyvj 1729 文艺平衡树 Time Limit: 10 Sec  Memory Limit: 128 MB Submit: 3313  Solved: 1883 [Submit][S ...

  3. [日常摸鱼]bzoj3224普通平衡树-Treap、Splay、01Trie、替罪羊树…

    http://www.lydsy.com/JudgeOnline/problem.php?id=3224 经典的平衡树模板题-各种平衡树好像都可以(黄学长之前好像还用vector卡过了这题) 所以这篇 ...

  4. 算法模板——平衡树Treap

    实现功能如下——1. 插入x数2. 删除x数(若有多个相同的数,因只删除一个)3. 查询x数的排名(若有多个相同的数,因输出最小的排名)4. 查询排名为x的数5. 求x的前驱(前驱定义为小于x,且最大 ...

  5. luoguP3369[模板]普通平衡树(Treap/SBT) 题解

    链接一下题目:luoguP3369[模板]普通平衡树(Treap/SBT) 平衡树解析 #include<iostream> #include<cstdlib> #includ ...

  6. 启发式合并&线段树合并/分裂&treap合并&splay合并

    启发式合并 有\(n\)个集合,每次让你合并两个集合,或询问一个集合中是否存在某个元素. ​ 我们可以用平衡树/set维护集合. ​ 对于合并两个\(A,B\),如果\(|A|<|B|\),那么 ...

  7. 普通平衡树Treap(含旋转)学习笔记

    浅谈普通平衡树Treap 平衡树,Treap=Tree+heap这是一个很形象的东西 我们要维护一棵树,它满足堆的性质和二叉查找树的性质(BST),这样的二叉树我们叫做平衡树 并且平衡树它的结构是接近 ...

  8. 2021.12.06 平衡树——Treap

    2021.12.06 平衡树--Treap https://www.luogu.com.cn/blog/HOJQVFNA/qian-xi-treap-ping-heng-shu 1.二叉搜索树 1.1 ...

  9. hiho #1325 : 平衡树·Treap

    #1325 : 平衡树·Treap 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Ho:小Hi,我发现我们以前讲过的两个数据结构特别相似. 小Hi:你说的是哪两个啊? ...

随机推荐

  1. delphi如何让程序最小化到任务栏(使用Shell_NotifyIcon API函数)

    现在很多的应用程序都有这样一种功能,当用户选择最小化窗口时,窗口不是象平常那样最小化到任务栏上,而是“最小化”成一个任务栏图标.象FoxMail 3.0 NetVampire 3.0等都提供了这样的功 ...

  2. Juery插件-- jquery.cookie.js

    1.引入jquery <script src="scripts/jquery-1.8.8.js" type="text/javascript">&l ...

  3. extjs grid renderer参数用法

    今天在导出EXT的二维时老是报错,追进去看是renderer : function(value)的参数不对,经过一番研究,未免以后遇到再次浪费时间,记录一下. var cm = new Ext.gri ...

  4. html body div height: 100%;

    最近做了测试 html{ height: 100%;//全部内容高度,包括滚动出现的内容 background-color:#000;} body{height: 100%;//只一页屏幕,用作滚动的 ...

  5. flex平分测试

    <!DOCTYPE html> <html> <head> <meta charset="utf-8" /> <title&g ...

  6. 如何刷新本地的DNS缓存?

    为了提高网站的访问速度,系统会在成功访问某网站后将该网站的域名.IP地址信息缓存到本地.下次访问该域名时直接通过IP进行访问.一些网站的域名没有变化,但IP地址发生变化,有可能因本地的DNS缓存没有刷 ...

  7. BZOJ_4154_[Ipsc2015]Generating Synergy_KDTree

    BZOJ_4154_[Ipsc2015]Generating Synergy_KDTree Description 给定一棵以1为根的有根树,初始所有节点颜色为1,每次将距离节点a不超过l的a的子节点 ...

  8. C++模板之可变模板参数

    可变模板参数---- C++11新特性 可变模板参数(variadic templates)是C++11新增的最强大的特性之一,它对参数进行了高度泛化,它能表示0到任意个数.任意类型的参数 由于可变模 ...

  9. springboot中使用@Value读取配置文件

    一.配置文件配置 直接配置 在src/main/resources下添加配置文件application.properties 例如修改端口号 #端口号 server.port=8089 分环境配置 在 ...

  10. layui 复选框checkbox 全选写法

    前语:本来我是不想写layui框架的博客的,有的时候数据经过layui渲染后原生的写法就取不到值了,一定要用它框架的写法,实在蛋疼,故写之,以后用到可以省点时间去度娘! HTML: <div i ...