LG3369 普通平衡树
题意
维护一些数,其中需要提供以下操作:
- 1.插入\(x\)
- 2.删除\(x\)(若有多个相同的数,只删除一个)
- 3.查询\(x\)的排名(排名定义为比当前数小的数的个数\(+1\))
- 4.查询排名为\(x\)的数
- 5.求最大的小于\(x\)数
- 6.求最小的大于\(x\)数
\(n \leq 100000\)
思路
这是一道\(treap\)模板
\(Treap=tree+heap\)
下图是一棵二叉排序树
性质:若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值;若它的右子树不空,则右子树上所有结点的值均大于它的根结点的值;它的左、右子树也分别为二叉排序树。
但是有一个问题,如果被卡之后很有可能变成一棵“链”搜索树。然后堆就现身了,可以随机另外附权值,在二叉排序树基础上,它还需要满足随机权值是小根堆,这时候就需要一个旋转操作。
这样子就可以让一些链重新压扁成树,而这些旋转操作,只会影响到三个点
void cal(int x){
size[x]=size[s[x][0]]+size[s[x][1]]+1;
}
void r(int &k,int p){
int t=s[k][p];
s[k][p]=s[t][!p],s[t][!p]=k;
cal(k),cal(t);
k=t;
}
然后就可以开始进行操作了,操作就是在普通二叉树上加上旋转操作
- 1.插入:找到这个点的位置,给每个点随机一个优先级,然后如果它的优先级要小于父节点,我们就把它旋转上去。
void ins(int &k,int x){
if (!k){
k=++cnt,w[k]=x,pos[k]=rand()*rand()%19620817,size[k]=1;
return;
}
size[k]++;
if (x<=w[k]){
ins(s[k][0],x);
if(pos[s[k][0]]<pos[k]) r(k,0);
}else{
ins(s[k][1],x);
if (pos[s[k][1]]<pos[k]) r(k,1);
}
}
- 2.删除:有点像堆的删除,先看看有没有。有的话找到这个点,将这个点优先级小的子节点旋转上来,一直将其旋到叶子,然后删除。
bool f(int k,int x){
if (w[k]==x) return 1;
if (!k) return 0;
if (w[k]>x) return f(s[k][0],x);
return f(s[k][1],x);
}
void del(int &k,int x){
if (x>w[k]) del(s[k][1],x);
else if (x<w[k]) del(s[k][0],x);
else{
if (s[k][0]*s[k][1]==0) {
k=s[k][0]+s[k][1];
return;
}
if (pos[s[k][0]]<pos[s[k][1]]) {r(k,0);del(s[k][1],x);}
else {r(k,1);del(s[k][0],x);}
}
cal(k);
}
- 3.查找\(x\)的排名:只要看左右子树的大小就可以了,但是要注意一点:因为查询的是最小的排名,所以当我们查到\(x\)的时候,不能直接返回,而要递归下去,直到到达叶节点为止
int find(int k,int x){
if (!k) return 1;
if (x<=w[k]) return find(s[k][0],x);
return size[s[k][0]]+1+find(s[k][1],x);
}
- 4.查询排名为\(x\)的数:直接左右子树判一判查下去
int find2(int k,int x){
if (x-1==size[s[k][0]]) return w[k];
if (x>size[s[k][0]]) return find2(s[k][1],x-size[s[k][0]]-1);
else return find2(s[k][0],x);
}
- 5.求最大的小于\(x\)数
- 6.求最小的大于\(x\)数
两个操作非常相似,如果找到了小于\(x\)的,则还要看一看左子树还有没比\(x\)小的,否则就去右子树。
int pre(int k,int x){
if (!k) return -INF;
if (w[k]<x) return max(pre(s[k][1],x),w[k]);
return pre(s[k][0],x);
}
int Next(int k,int x){
if (!k) return INF;
if (w[k]>x) return min(w[k],Next(s[k][0],x));
return Next(s[k][1],x);
}
完整版把上面所有的合在一起就好了
LG3369 普通平衡树的更多相关文章
- 普通平衡树 lg3369
在多次学习splay后,我终于理解并码出了整份代码 参考了https://tiger0132.blog.luogu.org/slay-notes的博客 具体实现原理在上面这篇博客和百度中可以查到,接下 ...
- LG3369 【模板】普通平衡树
题意 您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作: 插入x数 删除x数(若有多个相同的数,因只删除一个) 查询x数的排名(排名定义为比当前数小的数的个数+1.若有多个相 ...
- [BZOJ3223]Tyvj 1729 文艺平衡树
[BZOJ3223]Tyvj 1729 文艺平衡树 试题描述 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作:翻转一个区间,例如原有序序列是5 4 3 2 1,翻转区 ...
- [BZOJ3224]Tyvj 1728 普通平衡树
[BZOJ3224]Tyvj 1728 普通平衡树 试题描述 您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作:1. 插入x数2. 删除x数(若有多个相同的数,因只删除一个) ...
- BZOJ3223: Tyvj 1729 文艺平衡树 [splay]
3223: Tyvj 1729 文艺平衡树 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 3595 Solved: 2029[Submit][Sta ...
- [普通平衡树treap]【学习笔记】
3224: Tyvj 1728 普通平衡树 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 9046 Solved: 3840[Submit][Sta ...
- BZOJ 3224: Tyvj 1728 普通平衡树
3224: Tyvj 1728 普通平衡树 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 9629 Solved: 4091[Submit][Sta ...
- BZOJ 3223: Tyvj 1729 文艺平衡树
3223: Tyvj 1729 文艺平衡树 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 3628 Solved: 2052[Submit][Sta ...
- 【Splay】bzoj3223-Tyvj1729文艺平衡树
一.题目 Description 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作:翻转一个区间,例如原有序序列是5 4 3 2 1,翻转区间是[2,4]的话,结果是5 ...
随机推荐
- python学习第26天
自定义模块和包 软件开发规范
- 【原创】运维基础之Zabbix(1)简介、安装、使用
zabbix 4 官方:https://www.zabbix.com/ 一 简介 Monitor anythingSolutions for any kind of IT infrastructure ...
- java基础编程题练习(二)
1.回文数 思路一:使用java特有解法,将原数字以字符串存储,翻转后赋值给新的字符串变量,再使用equals与原字符串进行对比 import java.util.Scanner; public cl ...
- 03-django模型(1)
一.内容回顾 1.路由层 a.简单使用 b.有名分组 c.路由分发 d.反向解析 2.视图层 a.HttpRequest对象 常用的属性 常用方法 b.HttpResponse对象 响应三剑客 3.模 ...
- JDBC 返回主键
转载至:https://www.liyongzhen.com/ 上一节课里我们学习通过PreparedStatement对象执行带参数的查询SQL和修改SQL. 这节课我们学习使用 PreparedS ...
- MySQL This function has none of DETERMINISTIC, NO SQL...错误1418 的原因分析及解决方法 (转)
解决方法: 解决办法也有两种, 第一种是在创建子程序(存储过程.函数.触发器)时,声明为DETERMINISTIC或NO SQL与READS SQL DATA中的一个, 例如: CREATE DEFI ...
- CodeForces - 833B The Bakery
题解: 线段树经典应用 首先暴力$f[i][j]$表示考虑前i位分成j段的最大值 转移$f[k][j-1]+cost(k+1,i)$枚举k转移 不同数的经典套路就是从它到它前驱这一段 于是维护每个数前 ...
- spark Transformations算子
在java中,RDD分为javaRDDs和javaPairRDDs.下面分两大类来进行. 都必须要进行的一步. SparkConf conf = new SparkConf().setMaster(& ...
- 揭开JS闭包的面纱
今天看了关于js闭包方面的文章,还是有些云里雾里,对于一个菜鸟来说,学习闭包确实有一定的难度,不说别的,能够在网上找到一篇优秀的是那样的不易. 当然之所以闭包难理解,个人觉得是基础知识掌握的不牢,因为 ...
- NEO智能合约开发(一)不可能完成的任务
悬赏任务 兹有如下合约 public static object Main(string method, object[] args) { if (Runtime.Trigger == Trigger ...