P3380 【模板】二逼平衡树(树套树)
思路
若opt=1 则为操作1,之后有三个数l,r,k 表示查询k在区间[l,r]的排名
若opt=2 则为操作2,之后有三个数l,r,k 表示查询区间[l,r]内排名为k的数
若opt=3 则为操作3,之后有两个数pos,k 表示将pos位置的数修改为k
若opt=4 则为操作4,之后有三个数l,r,k 表示查询区间[l,r]内k的前驱
若opt=5 则为操作5,之后有三个数l,r,k 表示查询区间[l,r]内k的后继
树套树
区间操作可以用我们熟悉的线段树
线段树上每个点 维护对应区间[l,r]的一颗平衡树
线段树深度logn,每层要维护n个节点,所以treap的内存是nlogn
操作2的话,二分顺便用操作1check就好
二分的答案一定是在序列中的 ,复杂度\(log^{3}n\)(真的2b啊,什么复杂度)
其他具体操作就不啰嗦了(复杂度普遍\(log^{2}n\))
而n=5e4的时候\(log^{2}n\)和\(\sqrt{n}\)基本是相等的(分界线)
注意
线段树套平衡树
平衡树的种类也要选好
fhq-treap这种常数就比较大
普通treap就不错
splay也阔以
fhq-treap代码
#include <bits/stdc++.h>
#define FOR(i,a,b) for(int i=a;i<=b;++i)
using namespace std;
const int maxn=5e4+7;
const int inf=0x7fffffff;
inline int read() {
int x=0,f=1;char s=getchar();
for(;s>'9'||s<'0';s=getchar()) if(s=='-') f=-1;
for(;s>='0'&&s<='9';s=getchar()) x=x*10+s-'0';
return x*f;
}
int w[maxn];
int ch[maxn*40][2],val[maxn*40],pri[maxn*40],siz[maxn*40],cnt;
namespace fhq_treap {
void pushup(int x) {
siz[x]=siz[ch[x][0]]+siz[ch[x][1]]+1;
}
int new_node(int x) {
val[++cnt]=x;
pri[cnt]=rand();
siz[cnt]=1;
return cnt;
}
int merge(int x,int y) {
if(!x||!y) return x+y;
if(pri[x]<pri[y]) {
ch[x][1]=merge(ch[x][1],y);
pushup(x);return x;
} else {
ch[y][0]=merge(x,ch[y][0]);
pushup(y);return y;
}
}
void split(int now,int k,int &x,int &y) {
if(!now) x=y=0;
else {
if(val[now]<=k)
x=now,split(ch[now][1],k,ch[x][1],y);
else
y=now,split(ch[now][0],k,x,ch[y][0]);
pushup(now);
}
}
int find(int &rt,int a) {
int x,y;
split(rt,a-1,x,y);
int tmp=siz[x];
rt=merge(x,y);
return tmp;
}
int k_th(int now,int k) {
if(siz[now]<k||siz[now]==0||k==0) return 0x7fffffff;
while(233) {
if(siz[ch[now][0]]+1==k) return val[now];
if(siz[ch[now][0]]>=k) now=ch[now][0];
else k-=siz[ch[now][0]]+1,now=ch[now][1];
}
}
void insert(int &rt,int a) {
int x,y;
split(rt,a,x,y);
rt=merge(merge(x,new_node(a)),y);
}
void delet(int &rt,int a) {
int x,y,z;
split(rt,a,x,z);
split(x,a-1,x,y);
y=merge(ch[y][0],ch[y][1]);
rt=merge(merge(x,y),z);
}
int qq(int &rt,int a) {
int x,y,tmp;
split(rt,a-1,x,y);
tmp=k_th(x,siz[x]);
rt=merge(x,y);
return tmp==inf ? -inf : tmp;
}
int hj(int &rt,int a) {
int x,y,tmp;
split(rt,a,x,y);
tmp=k_th(y,1);
rt=merge(x,y);
return tmp;
}
}
struct node {
int l,r,rt;
}e[maxn<<2];
void build(int l,int r,int rt) {
e[rt].l=l,e[rt].r=r;
FOR(i,l,r) fhq_treap::insert(e[rt].rt,w[i]);
if(l==r) return;
int mid=(l+r)>>1;
build(l,mid,rt<<1);
build(mid+1,r,rt<<1|1);
}
void modify(int L,int k,int rt) {
fhq_treap::delet(e[rt].rt,w[L]);
fhq_treap::insert(e[rt].rt,k);
if(e[rt].l==e[rt].r) return;
int mid=(e[rt].l+e[rt].r)>>1;
if(L<=mid) modify(L,k,rt<<1);
else modify(L,k,rt<<1|1);
}
int find(int L,int R,int k,int rt) {
if(L<=e[rt].l&&e[rt].r<=R)
return fhq_treap::find(e[rt].rt,k);
int mid=(e[rt].l+e[rt].r)>>1,ans=0;
if(L<=mid) ans+=find(L,R,k,rt<<1);
if(R>mid) ans+=find(L,R,k,rt<<1|1);
return ans;
}
int qq(int L,int R,int a,int rt) {
if(L<=e[rt].l&&e[rt].r<=R) {
return fhq_treap::qq(e[rt].rt,a);
}
int mid=(e[rt].l+e[rt].r)>>1,ans=-inf;
if(L<=mid) ans=max(ans,qq(L,R,a,rt<<1));
if(R>mid) ans=max(ans,qq(L,R,a,rt<<1|1));
return ans;
}
int hj(int L,int R,int a,int rt) {
if(L<=e[rt].l&&e[rt].r<=R) {
return fhq_treap::hj(e[rt].rt,a);
}
int mid=(e[rt].l+e[rt].r)>>1,ans=inf;
if(L<=mid) ans=min(ans,hj(L,R,a,rt<<1));
if(R>mid) ans=min(ans,hj(L,R,a,rt<<1|1));
return ans;
}
int main() {
srand(time(NULL));
int n=read(),m=read();
FOR(i,1,n) w[i]=read();
build(1,n,1);
FOR(i,1,m) {
int opt=read();
if(opt==1) {
int l=read(),r=read(),k=read();
cout<<find(l,r,k,1)+1<<"\n";
} else
if(opt==2) {
int l=read(),r=read(),k=read();
int L=0,R=1e8+5,ans=0;
while(L<=R) {
int mid=(L+R)>>1;
if(find(l,r,mid,1)+1<=k) ans=mid,L=mid+1;
else R=mid-1;
}
cout<<ans<<"\n";
} else
if(opt==3) {
int pos=read(),k=read();
modify(pos,k,1);
w[pos]=k;
} else
if(opt==4) {
int l=read(),r=read(),k=read();
cout<<qq(l,r,k,1)<<"\n";
} else
if(opt==5) {
int l=read(),r=read(),k=read();
cout<<hj(l,r,k,1)<<"\n";
}
}
return 0;
}
P3380 【模板】二逼平衡树(树套树)的更多相关文章
- bzoj 3196 Tyvj 1730 二逼平衡树(线段树套名次树)
3196: Tyvj 1730 二逼平衡树 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1807 Solved: 772[Submit][Stat ...
- bzoj 3196/ Tyvj 1730 二逼平衡树 (线段树套平衡树)
3196: Tyvj 1730 二逼平衡树 Time Limit: 10 Sec Memory Limit: 128 MB[Submit][Status][Discuss] Description ...
- BZOJ3196 二逼平衡树 ZKW线段树套vector(滑稽)
我实在是不想再打一遍树状数组套替罪羊树了... 然后在普通平衡树瞎逛的时候找到了以前看过vector题解 于是我想:为啥不把平衡树换成vector呢??? 然后我又去学了一下ZKW线段树 就用ZKW线 ...
- BZOJ3196 二逼平衡树 【线段树套平衡树】
题目 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作: 1.查询k在区间内的排名 2.查询区间内排名为k的值 3.修改某一位值上的数值 4.查询k在区间内的前驱(前驱 ...
- BZOJ 3196 Tyvj 1730 二逼平衡树:线段树套splay
传送门 题意 给你一个长度为 $ n $ 有序数列 $ a $ ,进行 $ m $ 次操作,操作有如下几种: 查询 $ k $ 在区间 $ [l,r] $ 内的排名 查询区间 $ [l,r] $ 内排 ...
- [BZOJ3196] [Tyvj1730] 二逼平衡树(线段树 套 Splay)
传送门 至少BZOJ过了,其他的直接弃. 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作: 1.查询k在区间内的排名 2.查询区间内排名为k的值 3.修改某一位值上的 ...
- bzoj 3196 Tyvj 1730 二逼平衡树【线段树 套 splay】
四舍五入就是个暴力. 对于线段树的每个区间都开一棵按权值排序的splay 对于第二个操作,二分一下,每次查询mid的排名,复杂度 $ O(nlog(n)^{3}) $ 其余的操作都是$ O(nlog( ...
- P3380 【模板】二逼平衡树(树套树)(线段树套平衡树)
P3380 [模板]二逼平衡树(树套树) 前置芝士 P3369 [模板]普通平衡树 线段树套平衡树 这里写的是线段树+splay(不吸氧竟然卡过了) 对线段树的每个节点都维护一颗平衡树 每次把给定区间 ...
- 洛谷P3380 【模板】二逼平衡树(树套树)(线段树+树状数组)
P3380 [模板]二逼平衡树(树套树) 题目描述 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作: 查询k在区间内的排名 查询区间内排名为k的值 修改某一位值上的数 ...
- 洛谷 P3380 【模板】二逼平衡树(树套树)-线段树套splay
P3380 [模板]二逼平衡树(树套树) 题目描述 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作: 查询k在区间内的排名 查询区间内排名为k的值 修改某一位值上的数 ...
随机推荐
- Unity shader学习之Grab Pass实现玻璃效果
GrabPass可将当前屏幕的图像绘制在一张纹理中,可用来实现玻璃效果. 转载请注明出处:http://www.cnblogs.com/jietian331/p/7201324.html shader ...
- ModuleNotFoundError: No module named '_pydevd_bundle.pydevd_cython' error on debug
现象:pycharm调试代码出现错误:ModuleNotFoundError: No module named '_pydevd_bundle.pydevd_cython' error on debu ...
- 【转】robotFramework 与testlink集成
场景: robotframework 执行完用例之后,将执行结果报至testlink. 方案1: 通过TestLink-API-Python-client中的RF关键字 每条用例执行完成之后根据状态进 ...
- python 在列表,元组,字典变量前加*号
废话不说,直接上代码(可能很多人以前不知道有这种方法): a=[1,2,3]b=(1,2,3)c={1:"a",2:"b",3:"c"}pr ...
- 开发vue单页面Demo
第1步:安装webpack脚手架 npm install webpack -g (全局安装) (新电脑启动npm run dev版本报错,是因为webpack-server版本更新的问题,要安装pac ...
- golang学习笔记6 beego项目路由设置
golang学习笔记5 beego项目路由设置 前面我们已经创建了 beego 项目,而且我们也看到它已经运行起来了,那么是如何运行起来的呢?让我们从入口文件先分析起来吧: package main ...
- 浅析PAC,修改PAC文件及user-rule文件实现自动代理
浅析PAC,修改PAC文件及user-rule文件实现自动代理 代理自动配置(英语:Proxy auto-config,简称PAC)是一种网页浏览器技术,用于定义浏览器该如何自动选择适当的代理服务器来 ...
- js多个异步请求
一,两个(或多个)js异步并发执行,怎么在两个AJax异步操作之后执行一个新的操作 原题来自 ES6 方法 1.Promise 包装异步ajax操作,2.定义async 函数,3.用await等待pr ...
- scanf,fscanf,sscanf的区别
scanf是从文件中读 sscanf是从字符串中读 scanf是从键盘输入中读 fread :以字节位计算长度,按照指定的长度和次数读取数据,遇到结尾或完成指定长度读取后停止.fscanf :格式 ...
- @RefreshScope 的作用
让在application.properties里自定义的变量也能通过@Value 注解正常注入