题目描述

您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作(对于各个以往的历史版本):

  1. 插入x数

  2. 删除x数(若有多个相同的数,因只删除一个,如果没有请忽略该操作)

  3. 查询x数的排名(排名定义为比当前数小的数的个数+1。若有多个相同的数,因输出最小的排名)

  4. 查询排名为x的数

  5. 求x的前驱(前驱定义为小于x,且最大的数,如不存在输出-2147483647)

  6. 求x的后继(后继定义为大于x,且最小的数,如不存在输出2147483647)

和原本平衡树不同的一点是,每一次的任何操作都是基于某一个历史版本,同时生成一个新的版本。(操作3, 4, 5, 6即保持原版本无变化)

每个版本的编号即为操作的序号(版本0即为初始状态,空树)

输入输出格式

输入格式:

第一行包含一个正整数N,表示操作的总数。

接下来每行包含三个正整数,第 i 行记为 vi​,opti​,xi​。

vi​表示基于的过去版本号( 0≤vi​<i ),opti​ 表示操作的序号( 1≤opt≤6 ), xi​ 表示参与操作的数值

输出格式:

每行包含一个正整数,依次为各个3,4,5,6操作所对应的答案

输入输出样例

输入样例#1:

10
0 1 9
1 1 3
1 1 10
2 4 2
3 3 9
3 1 2
6 4 1
6 2 9
8 6 3
4 5 8
输出样例#1:

9
1
2
10
3

说明

数据范围:

对于28%的数据满足: 1≤n≤10

对于44%的数据满足: 1≤n≤2⋅102

对于60%的数据满足: 1≤n≤3⋅103

对于84%的数据满足: 1≤n≤105

对于92%的数据满足: 1≤n≤2⋅105

对于100%的数据满足:1≤n≤5⋅105 , −109≤xi​≤109

经实测,正常常数的可持久化平衡树均可通过,请各位放心

样例说明:

共10次操作,11个版本,各版本的状况依次是:

  1. [][]

  2. [9][9]

  3. [3, 9][3,9]

  4. [9, 10][9,10]

  5. [3, 9][3,9]

  6. [9, 10][9,10]

  7. [2, 9, 10][2,9,10]

  8. [2, 9, 10][2,9,10]

  9. [2, 10][2,10]

  10. [2, 10][2,10]

  11. [3, 9][3,9]

Solution:

  本题可持久化平衡树板子题(没啥好写的)。

  我们先用无旋treap打下普通平衡树那道板子题,那么可持久化无非是在之前版本的状态基础上每次新开节点记录新的状态就好了。

  于是只需要每次改为新建节点去merge,改为新建节点去split,记录每个版本的树根就好了(感觉很简单啊)。

代码:

/*Code by 520 -- 9.26*/
#include<bits/stdc++.h>
#define il inline
#define ll long long
#define RE register
#define For(i,a,b) for(RE int (i)=(a);(i)<=(b);(i)++)
#define Bor(i,a,b) for(RE int (i)=(b);(i)>=(a);(i)--)
using namespace std;
const int N=5e5+,inf=0x7fffffff;
int n,root[N],cnt;
struct node{
int ls,rs,siz,date,rnd;
}t[N*]; int gi(){
int a=;char x=getchar();bool f=;
while((x<''||x>'')&&x!='-') x=getchar();
if(x=='-') x=getchar(),f=;
while(x>=''&&x<='') a=(a<<)+(a<<)+(x^),x=getchar();
return f?-a:a;
} il int newnode(int v){
++cnt;
t[cnt].date=v,t[cnt].siz=,t[cnt].rnd=rand();
return cnt;
} il void up(int rt){t[rt].siz=t[t[rt].ls].siz+t[t[rt].rs].siz+;} int merge(int x,int y){
if(!x||!y) return x+y;
if(t[x].rnd<t[y].rnd){
int p=++cnt;t[p]=t[x];
t[p].rs=merge(t[p].rs,y);
up(p);
return p;
}
else {
int p=++cnt;t[p]=t[y];
t[p].ls=merge(x,t[p].ls);
up(p);
return p;
}
} void split(int rt,int k,int &x,int &y){
if(!rt) {x=y=;return;}
if(t[rt].date<=k){
x=++cnt;t[x]=t[rt];
split(t[x].rs,k,t[x].rs,y);
up(x);
}
else {
y=++cnt;t[y]=t[rt];
split(t[y].ls,k,x,t[y].ls);
up(y);
}
} void del(int &root,int v){
int x=,y=,z=;
split(root,v,x,z),split(x,v-,x,y);
y=merge(t[y].ls,t[y].rs);
root=merge(x,merge(y,z));
} void ins(int &root,int v){
int x=,y=; split(root,v,x,y);
root=merge(x,merge(newnode(v),y));
} il int kth(int rt,int v){
while(){
if(v<=t[t[rt].ls].siz) rt=t[rt].ls;
else if(v>t[t[rt].ls].siz+) v-=t[t[rt].ls].siz+,rt=t[rt].rs;
else return t[rt].date;
}
} il int id(int &root,int v){
int x=,y=,ans; split(root,v-,x,y);
ans=t[x].siz+;
root=merge(x,y);
return ans;
} il int pre(int &root,int v){
int x=,y=,ans; split(root,v-,x,y);
if(!x) return -inf;
ans=kth(x,t[x].siz);
root=merge(x,y);
return ans;
} il int suc(int &root,int v){
int x=,y=,ans; split(root,v,x,y);
if(!y) return inf;
ans=kth(y,);
root=merge(x,y);
return ans;
} int main(){
srand(time());
n=gi();
int v,opt,x;
For(i,,n){
v=gi(),opt=gi(),x=gi();
root[i]=root[v];
if(opt==) ins(root[i],x);
if(opt==) del(root[i],x);
if(opt==) printf("%d\n",id(root[i],x));
if(opt==) printf("%d\n",kth(root[i],x));
if(opt==) printf("%d\n",pre(root[i],x));
if(opt==) printf("%d\n",suc(root[i],x));
}
return ;
}

P3835 【模板】可持久化平衡树的更多相关文章

  1. luoguP3835 [模板]可持久化平衡树

    https://www.luogu.org/problemnew/show/P3835 因为博主精力和实力有限,学不懂 fhq treap 了,因此只介绍 leafy tree 解法 leafy tr ...

  2. 洛谷.3835.[模板]可持久化平衡树(fhq treap)

    题目链接 对每次Merge(),Split()时产生的节点都复制一份(其实和主席树一样).时间空间复杂度都为O(qlogq).(应该更大些 因为rand()?内存真的爆炸..) 对于无修改的操作实际上 ...

  3. Luogu P3835 【模板】可持久化平衡树(fhq Treap)

    P3835 [模板]可持久化平衡树 题意 题目背景 本题为题目普通平衡树的可持久化加强版. 题目描述 您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作(对于各个以往的历史版本 ...

  4. 洛谷P3835 【模板】可持久化平衡树

    题目背景 本题为题目 普通平衡树 的可持久化加强版. 数据已经经过强化 感谢@Kelin 提供的一组hack数据 题目描述 您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作( ...

  5. luogu P3919 [模板]可持久化数组(可持久化线段树/平衡树)(主席树)

    luogu P3919 [模板]可持久化数组(可持久化线段树/平衡树) 题目 #include<iostream> #include<cstdlib> #include< ...

  6. [Luogu 3835]【模板】可持久化平衡树

    Description 您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作(对于各个以往的历史版本): 插入x数 删除x数(若有多个相同的数,因只删除一个,如果没有请忽略该操作 ...

  7. LG3835 【模板】可持久化平衡树

    题意 您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作(对于各个以往的历史版本): 插入x数 删除x数(若有多个相同的数,因只删除一个,如果没有请忽略该操作) 查询x数的排名 ...

  8. 2021.07.02 P1383 高级打字机题解(可持久化平衡树)

    2021.07.02 P1383 高级打字机题解(可持久化平衡树) 分析: 从可以不断撤销并且查询不算撤销这一骚操作可以肯定这是要咱建一棵可持久化的树(我也只会建可持久化的树,当然,还有可持久化并查集 ...

  9. 可持久化Trie & 可持久化平衡树 专题练习

    [xsy1629]可持久化序列 - 可持久化平衡树 http://www.cnblogs.com/Sdchr/p/6258827.html [bzoj4260]REBXOR - Trie 事实上只是一 ...

随机推荐

  1. sql server sql查询数据库的表,字段,主键,自增,字段类型等信息

    1.查询数据表的属性(名称.说明.是否主键.数据类型.是否自增) SELECT t1.name columnName,case when t4.id is null then 'false' else ...

  2. c语言数字图像处理(十):阈值处理

    定义 全局阈值处理 假设某一副灰度图有如下的直方图,该图像由暗色背景下的较亮物体组成,从背景中提取这一物体时,将阈值T作为分割点,分割后的图像g(x, y)由下述公式给出,称为全局阈值处理 多阈值处理 ...

  3. WEB渗透测试基础工具

    代理查询网站:hidemyass(隐藏我的屁股) HTTrack:HTTrack是一个免费和易用的离线浏览工具(浏览器),它可以允许你下载整个WWW网站至本地目录,并且通过遍历网站目录获取HTML,图 ...

  4. 使用C#采集Shibor数据到Excel

    对Shibor的变化一直以来比较关注,正好最近学习了对html数据处理的一些知识,就打算拿来采集一些我需要的Shibor数据. 使用到的库 HttpAgilityPack 一个非常不错的html解析工 ...

  5. Spring AOP部分源码分析

    Spring源码流程分析-AOP相关 根据Spring源码整理,其中Calculator为自定义的实现方法. AnnotationConfigApplicationContext()加载配置类的流程 ...

  6. 2019第十届蓝桥杯C++B组题解(赛后重写的,不确保答案正确性,仅供参考)

    先说一下这次的感受吧,我们考场比较乱,开始比赛了,还有的电脑有故障,(向这些人发出同情),第一次认真参加比赛,真正比赛的时候感觉没有那么正式,很乱,各种小问题,(例如博主就没找到题目在哪里,找到后又不 ...

  7. Android工程导入Unity3D(避坑版)

    最近与各种牛逼的项目管理软件打交道,比如SourceTree,要看英文版的才看得懂,中文反而不会用!... 这篇博客适合没怎么接触过安卓的小伙伴们,网上也有很多相关的教程,但是大多都没有具体的操作或则 ...

  8. [salt] jinja模板中变量使用pillar的几种方法

    先转载下jinja模板中使用变量的方法,后文主要讲解pillar的变量使用方法 一.jinja模版的使用方法: 1.file状态使用template参数 - template:jinja 2.模版文件 ...

  9. 第九次作业psp

    psp 进度条 代码累积折线图 博文累积折线图 psp饼状图

  10. final发布--PSP Daily软件功能书(最终版)

    一.开发背景 你在完成了一周的软件工程作业后,需要提交一个PSP图表,里面有4项,如下所示: 1.本周PSP表格,包含每项任务的开始.中断.结束.最终时间,格式如下: 2.本周进度条,包含从开始到现在 ...