传送门

看了一圈,好像真的没什么用指针的呢。。

明明觉得指针很好看(什么??你说RE???听不见听不见)

其实我觉得用数组的话不RE直接WA调起来不是更困难嘛,毕竟通过gdb还可以知道哪里RE,WA就不知道咋回事了,是不是很有道理,虽然我还是调了几小时

我写的是fhq treap,核心是split和merge操作,思想高赞dalao都讲得很清楚,我语文弱渣就不班门弄斧了,主要是想提供一个指针版的参考吧QAQ

我真的是一整天都在搞分裂(split),有种要进入七月枪毙名单的赶脚,慌张.jpg

#include<bits/stdc++.h>
#define LL long long
#define fr(i,x,y) for(int i=(x);i<=(y);i++)
#define rf(i,x,y) for(int i=(x);i>=(y);i--)
#define frl(i,x,y) for(int i=(x);i<(y);i++)
using namespace std;
const int N=100005;
struct node{
int v,rnd,s,sz; //s表示权值为v的个数,sz表示子树size,然而我经常忘记s的存在,直接写成1,挂了好久
node* ch[2]; inline int cmp(int x){ return x>v; } //这其实是写旋转treap时留下的历史遗留问题= =无视吧 inline void maintain(){
sz=s;
if (ch[0]!=NULL) sz+=ch[0]->sz; //写指针一定要特别注意对NULL的判断
if (ch[1]!=NULL) sz+=ch[1]->sz;
} node(){
ch[0]=ch[1]=NULL;
rnd=rand();
}
}nd[N];
int tot;
node* rt;
int n; void read(int &x){ //读优一开始忘记负数了= =
char ch=getchar();x=0;int w=0;
for(;ch<'0'||ch>'9';ch=getchar()) if (ch=='-') w=1;
for(;ch>='0'&&ch<='9';ch=getchar()) x=(x<<3)+(x<<1)+ch-'0';
if (w) x=-x;
} inline int sss(node* &o){
return o==NULL?0:o->sz;
} node* &kth(node* &o,int k){
assert(o!=NULL);
int s=sss(o->ch[0]);
if (s+1<=k&&s+o->s>=k) return o;
if (s+1>k) return kth(o->ch[0],k);
return kth(o->ch[1],k-o->s-s);
} void split(node* o,node* &L,node* &r,int k){ //split <=k
if (o==NULL) return L=r=NULL,void();
if (o->v<=k){
split(o->ch[1],o->ch[1],r,k);
L=o;
} else{
split(o->ch[0],L,o->ch[0],k);
r=o;
}
o->maintain(); //要经常维护一下信息
} node* &merge(node* &L,node* &r){
if (L==NULL) return r;
if (r==NULL) return L;
if (L->rnd<r->rnd){
L->ch[1]=merge(L->ch[1],r);
L->maintain();
return L;
} else{
r->ch[0]=merge(L,r->ch[0]);
r->maintain();
return r;
}
} void add_node(int v){
node* L;
node* r;node* xx;
split(rt,L,r,v);
if (L!=NULL&&(xx=kth(L,L->sz))->v==v){ //如果存在这个数直接加个数
split(L,L,xx,v-1);
xx->s++;xx->sz++; //不要忘记加size
//rt=merge(L,r);
}else{
xx=&nd[++tot];
xx->s=xx->sz=1;xx->v=v;
}
rt=merge(merge(L,xx),r);
} void del_node(int v){
node *L,*r,*mid;
split(rt,L,r,v-1);
split(r,mid,r,v);
if (mid!=NULL&&mid->s>1) mid->s--,mid->sz--,r=merge(mid,r);
rt=merge(L,r);
} int rk(node* &o,int v){
assert(o!=NULL);
if (o->v==v) return sss(o->ch[0])+1;
if (v<o->v) return rk(o->ch[0],v);
else return rk(o->ch[1],v)+sss(o->ch[0])+o->s;
} int rkk(node* rt,int v){
node *L,*r;
split(rt,L,r,v-1);
int ans=sss(L)+1;
rt=merge(L,r);
return ans;
}
//rk和rkk都可以求rank,一个通过split一个通过size,好像rkk更快?感觉有点奇怪... int main(){
srand(19260817);
read(n);
int tp,x;
node *L,*r;
int s=0;
add_node(19260817);
fr(o,1,n){
read(tp);read(x);
if (tp==1) add_node(x);
if (tp==2) del_node(x);
if (tp==3) printf("%d\n",rk(rt,x)),s++;
if (tp==4) printf("%d\n",kth(rt,x)->v),s++;
if (tp==5){
split(rt,L,r,x-1);
printf("%d\n",kth(L,L->sz)->v),s++;
rt=merge(L,r);
}
if (tp==6){
split(rt,L,r,x);
printf("%d\n",kth(r,1)->v),s++;
rt=merge(L,r);
}
//if (s==670) printf("----%d %d %d\n",o,tp,x);
}
return 0;
}

非旋treap (fhq treap) 指针版的更多相关文章

  1. 非旋Treap——fhq treap

    https://www.luogu.org/problemnew/show/P3369 知识点:1.拆分split,合并merge 2.split,merge要点:通过传址调用来简便代码 3.记得ro ...

  2. 可持久化Treap(fhq Treap,非旋转式Treap)学习(未完待续)

    简介:     Treap,一种表现优异的BST 优势:     其较于AVL.红黑树实现简单,浅显易懂     较于Splay常数小,通常用于树套BST表现远远优于Splay     或许有人想说S ...

  3. 「学习笔记」 FHQ Treap

    FHQ Treap FHQ Treap (%%%发明者范浩强年年NOI金牌)是一种神奇的数据结构,也叫非旋Treap,它不像Treap zig zag搞不清楚(所以叫非旋嘛),也不像Splay完全看不 ...

  4. 2021.12.08 平衡树——FHQ Treap

    2021.12.08 平衡树--FHQ Treap http://www.yhzq-blog.cc/fhqtreapzongjie/ https://www.cnblogs.com/zwfymqz/p ...

  5. 可持久化treap(FHQ treap)

    FHQ treap 的整理 treap = tree + heap,即同时满足二叉搜索树和堆的性质. 为了使树尽可能的保证两边的大小平衡,所以有一个key值,使他满足堆得性质,来维护树的平衡,key值 ...

  6. 非旋 treap 结构体数组版(无指针)详解,有图有真相

    非旋  $treap$ (FHQ treap)的简单入门 前置技能 建议在掌握普通 treap 以及 左偏堆(也就是可并堆)食用本blog 原理 以随机数维护平衡,使树高期望为logn级别, FHQ  ...

  7. 非旋(fhq)Treap小记

    前置知识:二叉搜索树 以下摘自 ↑: 二叉搜索树每次操作访问O(深度)个节点. 在刻意构造的数据中,树的形态会被卡成一条链,于是复杂度爆炸 它的复杂度与直接暴力删除类似. 但二叉搜索树扩展性强.更复杂 ...

  8. 关于非旋FHQ Treap的复杂度证明

    非旋FHQ Treap复杂度证明(类比快排) a,b都是sort之后的排列(从小到大) 由一个排列a构造一颗BST,由于我们只确定了中序遍历=a,但这显然是不能确定一棵树的形态的. 由一个排列b构造一 ...

  9. [模板] 平衡树: Splay, 非旋Treap, 替罪羊树

    简介 二叉搜索树, 可以维护一个集合/序列, 同时维护节点的 \(size\), 因此可以支持 insert(v), delete(v), kth(p,k), rank(v)等操作. 另外, prev ...

随机推荐

  1. httprouter框架 (Gin使用的路由框架)

    之前在Gin中已经说到, Gin比Martini的效率高好多耶, 究其原因是因为使用了httprouter这个路由框架, httprouter的git地址是: httprouter源码. 今天稍微看了 ...

  2. Timed out after 30000 ms while waiting to connect

    今天使用mongo-java-drive写连接mongo的客户端,着实被上面那个错坑了一把.回顾一下解决过程: 报错: com.mongodb.MongoTimeoutException: Timed ...

  3. MySQLdb操作数据库

    堡垒机前戏 开发堡垒机之前,先来学习Python的paramiko模块,该模块机遇SSH用于连接远程服务器并执行相关操作 SSHClient 用于连接远程服务器并执行基本命令 基于用户名密码连接: + ...

  4. SQL SERVER 数据库中查看文本字段中的数据长度LEN() 函数的使用方法

    SQL LEN() 语法 SELECT LEN(column_name) FROM table_name Id LastName FirstName Address City 1 Adams John ...

  5. 2019滴滴php面试总结 (包含面试题解析)

    2019滴滴java面试总结  (包含面试题) 本人6年开发经验.今年年初找工作,在互联网寒冬下成功拿到阿里巴巴.今日头条.滴滴等公司offer,岗位是既有php也有Java后端开发,最终选择去了滴滴 ...

  6. VMware15.5版本下安装Windows_Server_2008_R2

    一.新建虚拟机 第一步:打开VMware15.5虚拟机,在欢迎界面点击新建虚拟机: 第二步:选择典型(推荐)选项-->适用于新手,单击下一步: 第三步:选定最后一项稍后安装操作系统,单击下一步: ...

  7. Python_函数传参

    关于函数中传递参数的相关知识 其中 万能参数 第一次听说 但感觉用处不大 后面用到再详细整理

  8. JBOSS中间件漏洞总汇复现

    JBOSS中间件漏洞总汇复现 JBoss JMXInvokerServlet 反序列化漏洞 漏洞复现 直接使用docker搭建的漏洞环境. 环境搭建完成后,直接使用工具检测即可:工具下载地址https ...

  9. 网页背景H5视频自动播放---PC端、移动端兼容问题完美解决方案(IOS、安卓、微信端)

    最近公司官网需要使用视频当做banner背景且自动播放,并且因为是官网需要做到PC端和移动端都可以适配兼容,这些问题很是头疼: 兵来将挡,水来土掩,进过查阅相关技术资料,现已完美兼容PC端和移动端.下 ...

  10. 扩展阿里p3c实现自定义代码规范检查

     前段时间fastjson报出了漏洞,只要打开setAutoType特性就会存在风险,自己测试环境的一个项目被揪出来了-_-!.虽然改动很小,但就是觉得憋屈.fastjson还是挺好的,想着禁用的话太 ...