\(\text{fhq-treap}\)总结

又名范浩强\(\text{treap}\),是一种无旋\(\text{treap}\)。其原理同\(\text{treap}\)一样都是通过维护一个随机堆来避免退化为单链的情况,但是无需旋转\(\text{rotate}\),而是通过分裂\(\text{split}\)完成操作。其代码极度简洁,极度舒适,反正拯救了我的平衡树。

基本操作

\(\text{split}\)

分裂为左右两个子树\(x,y\),子树\(x\)内的数均\(\le k\),而子树\(y\)内的数$ >k\(。(这里是按权值分裂,还有一种按子树大小分裂的,即子树\)x\(大小为\)k$,维护区间翻转的时候需要用)

当前节点值\(\le k\)时,说明当前节点及其左子树肯定都是\(\le k\),当前要找的\(x\)子树树根就是当前节点,而\(y\)子树树根不确定。而且对于其右子树是否均\(>k\)我们并不确定,因为可能右儿子的左子树中可能还有\(\le k\)的,所以我们还要递归找下去。

void split(int cur, int k, int &x, int &y){
if(cur==0){x=y=0;return;}
if(tre[cur].val<=k){
x=cur;
split(tre[cur].sr, k, tre[cur].sr, y); // 递归找其右儿子
}else{
y=cur;
split(tre[cur].sl, k, x, tre[cur].sl); // 递归找其左儿子
}
update(cur); // 更新子树大小
}

\(\text{merge}\)

合并两个子树\(x,y\),默认子树\(x\)均小于子树\(y\)中的数

合并子树的同时利用随机权值维护一个堆(这里是小根堆)。维护堆的同时维护\(\text{BST}\),让小于等于自己的数成为左儿子,大于自己的数成为右儿子。

int merge(int x, int y){
if(!x||!y) return x|y;
if(tre[x].rnd<tre[y].rnd){
tre[x].sr=merge(tre[x].sr, y);
update(x);
return x;
}else{
tre[y].sl=merge(x, tre[y].sl);
update(y);
return y;
}
}

\(\text{new_nod}\)

新建一个节点并返回其编号。

inline int new_nod(int val){
++tot;
tre[tot].val=val;
tre[tot].rnd=rand();
tre[tot].sz=1;
return tot;
}

\(\text{insert}\)

插入数值\(val\)

inline void insert(int val){
split(root, val, x, y);
root=merge(merge(x, new_nod(val)), y);
}

\(\text{del}\)

删除数值\(val\)

通过两次分裂子树,定位到全为\(val\)子树(节点)的位置,然后直接将这个全为\(val\)的子树直接删除。

inline void del(int val){
split(root, val, x, z);
split(x, val-1, x, y);
y=merge(tre[y].sl, tre[y].sr);
root=merge(merge(x,y), z);
}

\(\text{rank}\)

获取数值\(val\)的排名

inline int get_rank(int val){
split(root, val-1, x, y);
int res=tre[x].sz+1;
root=merge(x,y);
return res;
}

\(\text{kth}\)

获得第\(k\)大的数值

inline int get_kth(int cur, int k){
while(1){
if(k<=tre[tre[cur].sl].sz) cur=tre[cur].sl;
else if(k==tre[tre[cur].sl].sz+1) return cur;
else k-=tre[tre[cur].sl].sz+1, cur=tre[cur].sr;
}
}

\(\text{pre}\)

前驱,注意存在多个\(val\)数值的情况

inline int get_pre(int val){
split(root, val-1, x, y);
int res=get_kth(x, tre[x].sz);
root=merge(x,y);
return res;
}

\(\text{nxt}\)

后驱

inline int get_nxt(int val){
split(root, val, x, y);
int res=get_kth(y, 1);
root=merge(x,y);
return res;
}

普通平衡树

  1. 插入x数
  2. 删除x数(若有多个相同的数,因只删除一个)
  3. 查询x数的排名(排名定义为比当前数小的数的个数+1。若有多个相同的数,因输出最小的排名)
  4. 查询排名为x的数
  5. 求x的前驱(前驱定义为小于x,且最大的数)
  6. 求x的后继(后继定义为大于x,且最小的数)

完整代码:

#include <cstdio>
#include <cstdlib>
#define MAXN 100010
using namespace std;
struct nod{
int sl,sr;
int rnd,val,sz;
} tre[MAXN];
inline void update(int x){
tre[x].sz=tre[tre[x].sl].sz+tre[tre[x].sr].sz+1;
}
int merge(int x, int y){
if(!x||!y) return x|y;
if(tre[x].rnd<tre[y].rnd){
tre[x].sr=merge(tre[x].sr, y);
update(x);
return x;
}else{
tre[y].sl=merge(x, tre[y].sl);
update(y);
return y;
}
}
void split(int cur, int k, int &x, int &y){
if(cur==0){
x=y=0;
return;
}
if(tre[cur].val<=k){
x=cur;
split(tre[cur].sr, k, tre[cur].sr, y);
}else{
y=cur;
split(tre[cur].sl, k, x, tre[cur].sl);
}
update(cur);
}
int tot,root,x,y,z;
inline int new_nod(int val){
++tot;
tre[tot].val=val;
tre[tot].rnd=rand();
tre[tot].sz=1;
return tot;
}
inline void insert(int val){
split(root, val, x, y);
root=merge(merge(x, new_nod(val)), y);
}
inline void del(int val){
split(root, val, x, z);
split(x, val-1, x, y);
y=merge(tre[y].sl, tre[y].sr);
root=merge(merge(x,y), z);
}
inline int get_rank(int val){
split(root, val-1, x, y);
int res=tre[x].sz+1;
root=merge(x,y);
return res;
}
inline int get_kth(int cur, int k){
while(1){
if(k<=tre[tre[cur].sl].sz) cur=tre[cur].sl;
else if(k==tre[tre[cur].sl].sz+1) return cur;
else k-=tre[tre[cur].sl].sz+1, cur=tre[cur].sr;
}
}
inline int get_pre(int val){
split(root, val-1, x, y);
int res=get_kth(x, tre[x].sz);
root=merge(x,y);
return res;
}
inline int get_nxt(int val){
split(root, val, x, y);
int res=get_kth(y, 1);
root=merge(x,y);
return res;
}
int t;
int main()
{
srand((unsigned)19270817);
scanf("%d", &t);
while(t--){
int opt,a;
scanf("%d %d", &opt, &a);
if(opt==1) insert(a);
else if(opt==2) del(a);
else if(opt==3) printf("%d\n", get_rank(a));
else if(opt==4) printf("%d\n", tre[get_kth(root, a)].val);
else if(opt==5) printf("%d\n", tre[get_pre(a)].val);
else printf("%d\n", tre[get_nxt(a)].val);
}
return 0;
}

文艺平衡树

支持区间翻转,多次区间翻转,最后询问最终序列

这里我们就按树大小分裂子树,每次翻转区间\([l,r]\)时,通过两次分裂找出包含区间\([l,r]\)的树,然后用懒标记标记一下(\(change\)表示是否旋转当前树所代表的区间)。每次操作或查询前下放标记即可。

#include <cstdio>
#include <algorithm>
#include <cstdlib>
#define MAXN 100010
using namespace std;
struct nod{
int l,r,val,rnd,sz;
bool change;
} tre[MAXN];
int rot,tot;
inline void push_down(int x){
swap(tre[x].l, tre[x].r);
if(tre[x].l) tre[tre[x].l].change=!tre[tre[x].l].change;
if(tre[x].r) tre[tre[x].r].change=!tre[tre[x].r].change;
tre[x].change=0;
}
inline void update(int x){
tre[x].sz=tre[tre[x].l].sz+tre[tre[x].r].sz+1;
}
inline int new_nod(int val){
++tot;
tre[tot].val=val;
tre[tot].rnd=rand();
tre[tot].sz=1;
return tot;
}
int merge(int x, int y){
if(x==0||y==0) return x|y;
if(tre[x].rnd<tre[y].rnd){
if(tre[x].change) push_down(x);
tre[x].r=merge(tre[x].r, y);
update(x);
return x;
}else{
if(tre[y].change) push_down(y);
tre[y].l=merge(x, tre[y].l);
update(y);
return y;
}
}
void split(int cur, int k, int &x, int &y){
if(!cur){x=y=0;return;}
if(tre[cur].change) push_down(cur);
if(tre[tre[cur].l].sz<k){
x=cur;
split(tre[cur].r, k-tre[tre[cur].l].sz-1, tre[cur].r, y);
}else{
y=cur;
split(tre[cur].l, k, x, tre[cur].l);
}
update(cur);
}
void change(int l, int r){
int x,y,z;
split(rot,l-1,x,y);
split(y,r-l+1,y,z);
tre[y].change=!tre[y].change;
rot=merge(merge(x,y),z);
}
void dfs(int x){
if(!x) return;
if(tre[x].change) push_down(x);
dfs(tre[x].l);
printf("%d ", tre[x].val);
dfs(tre[x].r);
}
int n,m;
int main()
{
srand((unsigned)19270817);
scanf("%d %d", &n, &m);
for(int i=1;i<=n;++i) rot=merge(rot, new_nod(i));
while(m--){
int l,r;
scanf("%d %d", &l, &r);
change(l, r);
}
dfs(rot);
return 0;
}

随机推荐

  1. Spark 系列(六)—— 累加器与广播变量

    一.简介 在 Spark 中,提供了两种类型的共享变量:累加器 (accumulator) 与广播变量 (broadcast variable): 累加器:用来对信息进行聚合,主要用于累计计数等场景: ...

  2. redis数据结构和常用命令

    redis常用数据结构 String 最简单的K_V,value可以是数字或者字符串,使用场景:微博数.普通计数,命令:get set incr(加1) decr(减1) mget(获取多个值),se ...

  3. PDA日常问题

    一.连接网络异常 1.摩托摩拉3190连接wifi时报错,提示:scan error adapter unavailable 确认网卡是不是禁用状态,CE是右下角有个蓝色的图,上面有个X,点一下,然后 ...

  4. c# $和@ 简化字符串格式化拼接

    int age=18; Console.WriteLine($"XiaoMing is \"{age}\" {{ years}} old"); Console. ...

  5. Ajax + PHP 的用法以及遇见的问题

    由于自己是个php小白,所以新知识点都要自己去不断的试验和摸索. 分享下自己用php + ajax交互的用法和问题. 前端代码: $.ajax({ type: "POST", da ...

  6. 服务接口,选择rpc还是http?

    从通信内容/功能上看 http应用于web环境,rpc应用于分布式调度从功能上看没有太大区别,很多情况下rpc与消息中间件结合通信实现分布式调度 从用法上看两者都是c/s结构,无太大区别 从实现上看类 ...

  7. Linux Shell 小数比较

    #!/bin/bash #######expr 方法是错误的,在比较相同位数时可以,当位数不同就会出错,如100.00>70.00就会得出错误的结果 a=123b=123c=99.99rat=` ...

  8. nginx反向代理实现均衡负载及调度方法

    http upstream配置参数: ngx_http_upstream_module模块将多个服务器定义成服务器组,而由proxy_pass, fastcgi_pass等指令进行引用 upstrea ...

  9. Linux命令——systemctl

    前言 systemctl本身的意义并不仅仅是一个命令那么简单,他标志着SysV时代的终结,Systemd时代的开始.CentOS 7.X系列已经抛弃SysV,全面拥抱Systemd这个init sys ...

  10. 手写一个简易版Tomcat

    前言 Tomcat Write MyTomcat Tomcat是非常流行的Web Server,它还是一个满足Servlet规范的容器.那么想一想,Tomcat和我们的Web应用是什么关系? 从感性上 ...