.....好吧....最后一篇学习笔记的flag它倒了.....

好吧,这篇笔记也鸽了好久好久了...

比赛前刷模板,才想着还是补个坑吧...

FHQ,这个神仙(范浩强大佬),发明了这个神仙的数据结构,

首先,本篇博客使用洛谷普通平衡树为背景,即

  • 查找前驱
  • 查找后记
  • 查找kth的数
  • 查找k的排名
  • 插入一个数
  • 删除一个数

FHQ treap,是一个treap,它还是和treap一样,是tree+heap,所以它也有一个键值维护堆的性质。

它可以干任何treap和Splay能干的事。

它的实现主要由两个函数实现:

merge:把两棵树合并成一棵

split:把树分割成两棵

在这里介绍两个函数的实现方法:

merge

可以看到,它把两棵树合并了起来,但是并不是简单地接起来,而是打散,重新组合。

代码:

int merge(int x,int y)//把xy为根的两棵子树给合并
{
if(!x||!y)//如果一边没了
return x+y;//就返回
if(t[x].key<t[y].key)//维护key值,如果x的key值小于y的k值
{
t[x].son[]=merge(t[x].son[],y);//说明此时一定不符合堆性质,把x的右儿子和y合并
update(x);//更新相关变量
return x;//返回根节点
}
else
{
t[y].son[]=merge(x,t[y].son[]);//同上
update(y);
return y;
}
}

通过这样一个递归,不断拆分节点&&合并的过程中,就建立了一棵新树。

split:

从上图可得:(把树从5分开)

split的过程就是把树拆分成左右树,左树所有节点权值都小于k,右树的节点权值都大于k。

怎么实现呢?

代码:

void split(int now,int k,int &x,int &y)//把一棵树now给从k分割成x和y
{
if(!now) x=y=;//如果没有了,就返回
else
{
if(t[now].v<=k) //如果当前点的权值小于k,它应该在左子树
{
x=now;//更新
split(t[now].son[],k,t[now].son[],y);分割右儿子,找一个可能的更大的
}
else//同上
{
y=now;
split(t[now].son[],k,x,t[now].son[]);
}
update(now);
}
}

这样,我们就可以干以上的事了。

前置:查找kth

因为建立的事一个二叉查找树,所以还是可以像遍历二叉查找树那样查找kth的。

代码十分简单

int kth(int now,int k)
{
while()
{
if(k<=t[t[now].son[]].size)
now=t[now].son[];
else
{
if(k==t[t[now].son[]].size+)
return now;
else
{
k-=t[t[now].son[]].size+;
now=t[now].son[];
}
}
}
}

然后就可以A掉普通平衡树了。

插入新节点:首先暴力新建一个节点

int new_node(int k)
{
tot++;
t[tot].size=;
t[tot].v=k;
t[tot].key=rand();
return tot;
}

然后把树从k地方断开,把新节点看做一棵树,把它和上下树合在一起就行了

split(rt,a,x,y);
rt=merge(merge(x,new_node(a)),y);

删除节点:

把树从k断开,然后把左树从k-1断开,然后把上下树给合并,把k节点扔了就行了

split(rt,a,x,z);
split(x,a-,x,y);
y=merge(t[y].son[],t[y].son[]);
rt=merge(merge(x,y),z);

查找排名:

把树从k分开,则k所在的数的size即使排名

split(rt,a-,x,y);
printf("%d\n",t[x].size+);
rt=merge(x,y);

查找kth:

直接用kth函数即可

printf("%d\n",t[kth(rt,a)].v);

前驱:

把树从k分开,则size-1大小的那个kth点就是前驱

split(rt,a-,x,y);
printf("%d\n",t[kth(x,t[x].size)].v);
rt=merge(x,y);

后继:同上

split(rt,a,x,y);
printf("%d\n",t[kth(y,)].v);
rt=merge(x,y);

完整高清无码代码:

#include<bits/stdc++.h>
using namespace std;
const int maxn=1e6+;
struct tree
{
int son[],v,key,size;
}t[maxn];
int tot=,rt=;
void update(int p)
{
t[p].size=t[t[p].son[]].size+t[t[p].son[]].size+;
}
int new_node(int k)
{
tot++;
t[tot].size=;
t[tot].v=k;
t[tot].key=rand();
return tot;
}
int merge(int x,int y)//o?2¢ò?x£?y?a?ùμ?á???×óê÷
{
if(!x||!y)
return x+y;
if(t[x].key<t[y].key)
{
t[x].son[]=merge(t[x].son[],y);
update(x);
return x;
}
else
{
t[y].son[]=merge(x,t[y].son[]);
update(y);
return y;
}
}
void split(int now,int k,int &x,int &y)//ò?è¨?μk·?à?nowê÷3éx,y
{
if(!now) x=y=;
else
{
if(t[now].v<=k) //°??ùóDD?óúkμ?è¨?μμ??úμ?·?μ?ò???ê÷?D
{
x=now;
split(t[now].son[],k,t[now].son[],y);
}
else
{
y=now;
split(t[now].son[],k,x,t[now].son[]);
}
update(now);
}
}
int kth(int now,int k)
{
while()
{
if(k<=t[t[now].son[]].size)
now=t[now].son[];
else
{
if(k==t[t[now].son[]].size+)
return now;
else
{
k-=t[t[now].son[]].size+;
now=t[now].son[];
}
}
}
}
int x,y,z,n;
int main()
{
srand((unsigned)time(NULL));
scanf("%d",&n);
int flag,a,b,c;
for(int i=;i<=n;i++)
{
scanf("%d",&flag);
scanf("%d",&a);
if(flag==)
{
split(rt,a,x,y);
rt=merge(merge(x,new_node(a)),y);
}
if(flag==)
{
split(rt,a,x,z);
split(x,a-,x,y);
y=merge(t[y].son[],t[y].son[]);
rt=merge(merge(x,y),z);
}
if(flag==)
{
split(rt,a-,x,y);
printf("%d\n",t[x].size+);
rt=merge(x,y);
}
if(flag==)
{
printf("%d\n",t[kth(rt,a)].v);
}
if(flag==)
{
split(rt,a-,x,y);
printf("%d\n",t[kth(x,t[x].size)].v);
rt=merge(x,y);
}
if(flag==)
{
split(rt,a,x,y);
printf("%d\n",t[kth(y,)].v);
rt=merge(x,y);
}
}
return ;
}

FHQ treap学习(复习)笔记的更多相关文章

  1. fhq treap 学习笔记

    序 今天心血来潮,来学习一下fhq treap(其实原因是本校有个OIer名叫fh,当然不是我) 简介 fhq treap 学名好像是"非旋转式treap及可持久化"...听上去怪 ...

  2. 强化学习复习笔记 - DEEP

    Outline 激活函数 使用逼近器的特点: 较少数量的参数表达复杂的函数 (计算复杂度) 对一个权重的调整可以影响到很多的点 (泛化能力) 多种特征表示和逼近器结构 (多样性) 激活函数 Sigmo ...

  3. 「FHQ Treap」学习笔记

    话说天下大事,就像fhq treap —— 分久必合,合久必分 简单讲一讲.非旋treap主要依靠分裂和合并来实现操作.(递归,不维护fa不维护cnt) 合并的前提是两棵树的权值满足一边的最大的比另一 ...

  4. 「学习笔记」 FHQ Treap

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

  5. Fhq Treap [FhqTreap 学习笔记]

    众所周知 Fhq Treap 是 fhq 神仙研究出来的平衡树- 具体实现 每个点实现一个 \(\text{rnd}\) 表示 rand 的值 为什么要 rand 呢 是为了保证树高为 \(\log ...

  6. FFT/NTT复习笔记&多项式&生成函数学习笔记Ⅲ

    第三波,走起~~ FFT/NTT复习笔记&多项式&生成函数学习笔记Ⅰ FFT/NTT复习笔记&多项式&生成函数学习笔记Ⅱ 单位根反演 今天打多校时 1002 被卡科技了 ...

  7. treap学习笔记

    treap是个很神奇的数据结构. 给你一个问题,你可以解决它吗? 这个问题需要treap这个数据结构. 众所周知,二叉查找树的查找效率低的原因是不平衡,而我们又不希望用各种奇奇怪怪的旋转来使它平衡,那 ...

  8. fhq treap抄袭笔记

    目录 碎碎念 点一下 注意!!! 模板 fhq treap 碎碎念 我咋感觉合并这么像左偏树呢 ps:难道你们的treap都是小头堆的吗 fhq真的是神人 现在看以前学的splay是有点恶心,尤其是压 ...

  9. fhq treap最终模板

    新学习了fhq treap,厉害了 先贴个神犇的版, from memphis /* Treap[Merge,Split] by Memphis */ #include<cstdio> # ...

随机推荐

  1. 从零开始入门 K8s | 可观测性:监控与日志

    作者 | 莫源  阿里巴巴技术专家 一.背景 监控和日志是大型分布式系统的重要基础设施,监控可以帮助开发者查看系统的运行状态,而日志可以协助问题的排查和诊断. 在 Kubernetes 中,监控和日志 ...

  2. java之ReentrantLock详解

    前言 如果一个代码块被synchronized修饰了,当一个线程获取了相应的锁,并执行该代码块时,其他线程便只能一直等待,等待获取锁的释放,现在有这么一种情况,这个获取锁的线程由于要等待IO或者其他原 ...

  3. for循环练习题1——水仙花数

    /*输出所有的水仙花数,所谓水仙花数是指一个3位数,其各个位上数 字立方和等于其本身. 例如: 153 = 1*1*1 + 3*3*3 + 5*5*5 */class ForTest3{ public ...

  4. Java源码解析|HashMap的前世今生

    HashMap的前世今生 Java8在Java7的基础上,做了一些改进和优化. 底层数据结构和实现方法上,HashMap几乎重写了一套 所有的集合都新增了函数式的方法,比如说forEach,也新增了很 ...

  5. CS184.1X 计算机图形学导论L3V2和L3V3(部分)

    组合变换 连接矩阵的优点是可以使用这些矩阵单独操作. 多个变换依然是一个矩阵. 连接矩阵不可交换,因为矩阵乘法不具有交换性. X3=RX2 X2=SX1 X3=R(SX1)=(RS)X1 X3≠SRX ...

  6. [Noip2007] 字符串的展开

    题目描述 在初赛普及组的“阅读程序写结果”的问题中,我们曾给出一个字符串展开的例子:如果在输入的字符串中,含有类似于“d-h”或者“4-8”的字串,我们就把它当作一种简写,输出时,用连续递增的字母或数 ...

  7. 【Labview入门】子VI的调用

    labview版本:Labview2015 Labview里面也可以向C语言等变成语言一样调用子程序,本文带你一步一步实现子VI的创建与调用. 第一步 写好程序 首先我们写个简单的加法程序: 我们的目 ...

  8. std::multimap

    标准库还定义了一个 multimap 容器,它与 map 类似,所不同的是它允许重复键. 成员函数 insert() make_pair() 辅助函数来完成此任务. find(k) 返回指向第一个与键 ...

  9. POJ 3020 Antenna Placement(二分图 匈牙利算法)

    题目网址:  http://poj.org/problem?id=3020 题意: 用椭圆形去覆盖给出所有环(即图上的小圆点),有两种类型的椭圆形,左右朝向和上下朝向的,一个椭圆形最多可以覆盖相邻的两 ...

  10. 四jmeter脚本开发

    4.1工作区介绍 4.2http协议录制 a.使用badboy进行录制(目前公司测试系统使用badboy界面不友好,暂时不用,之后再补充) b.jmeter代理服务器进行录制 (1)浏览器代理配置(以 ...