洛谷P3834 [模板]可持久化线段树1(主席树) [主席树]
可持久化线段树1(主席树)
题目背景
这是个非常经典的主席树入门题——静态区间第K小
数据已经过加强,请使用主席树。同时请注意常数优化
题目描述
如题,给定N个正整数构成的序列,将对于指定的闭区间查询其区间内的第K小值。
输入输出格式
输入格式:
第一行包含两个正整数N、M,分别表示序列的长度和查询的个数。
第二行包含N个正整数,表示这个序列各项的数字。
接下来M行每行包含三个整数 $l,r,k$ , 表示查询区间 $[l,r]$ 内的第k小值。
输出格式:
输出包含k行,每行1个正整数,依次表示每一次查询的结果
输入输出样例
5 5
25957 6405 15770 26287 26465
2 2 1
3 4 1
4 5 1
1 2 2
4 4 1
6405
15770
26287
25957
26287
说明
数据范围:
对于20%的数据满足: $1 \leq N, M \leq 10$
对于50%的数据满足: $1 \leq N, M \leq 10^3$
对于80%的数据满足: $1 \leq N, M \leq 10^5$
对于100%的数据满足: $1 \leq N, M \leq 2\cdot 10^5$
对于数列中的所有数 $a_i$ ,均满足 $-{10}^9 \leq a_i \leq {10}^9$
样例数据说明:
N=5,数列长度为5,数列从第一项开始依次为 $[25957,6405,15770,26287,26465]$
第一次查询为 $[2,2]$ 区间内的第一小值,即为6405
第二次查询为 $[3,4]$ 区间内的第一小值,即为15770
第三次查询为 $[4,5]$ 区间内的第一小值,即为26287
第四次查询为 $[1,2]$ 区间内的第二小值,即为25957
第五次查询为 $[4,4]$ 区间内的第一小值,即为26287
分析:
主席树入门题。
一直说要学习主席树来的,但是直到今天才实现。主席树的具体思想蒟蒻就不再赘述,只说下$k$小值查询如何实现。查询静态$k$小值的时候在主席树中存储的是一个权值,也就是说我们把这棵主席树当做一颗权值树。现将数据离散,然后按照原顺序依次插入线段树中,从根节点开始往下直到找到第该元素排名个数个的线段树中节点的位置,将沿途经过的每个节点的记录的值+1。这样的话根据线段树中的权值就可以用一种类似于平衡树查询$k$小值的方法来查询。具体实现看代码。
Code:
//It is made by HolseLee on 29th July 2018
//Luogu.org P3834
#include<bits/stdc++.h>
using namespace std; const int N=2e5+;
int n,m,tot,cnt,a[N],b[N],rk[N],rt[N];
struct President_tree{
int ls,rs,sum;
}t[N*]; int read()
{
char ch=getchar();int num=;bool flag=false;
while(ch<''||ch>''){if(ch=='-')flag=true;ch=getchar();}
while(ch>=''&&ch<=''){num=(num<<)+(num<<)+ch-'';ch=getchar();}
return flag?-num:num;
} inline void build(int l,int r,int &node)
{
node=++cnt;
if(l==r)return ;
int mid=(l+r)>>;
build(l,mid,t[node].ls);
build(mid+,r,t[node].rs);
} inline void update(int l,int r,int &node,int last,int tar)
{
node=++cnt;t[node]=t[last];++t[node].sum;
if(l==r)return;
int mid=(l+r)>>;
if(tar<=mid)update(l,mid,t[node].ls,t[last].ls,tar);
else update(mid+,r,t[node].rs,t[last].rs,tar);
} inline int quary(int l,int r,int node,int last,int tar)
{
if(l==r)return a[l];
int now=t[t[node].ls].sum-t[t[last].ls].sum,mid=(l+r)>>;
if(tar<=now)return quary(l,mid,t[node].ls,t[last].ls,tar);
else return quary(mid+,r,t[node].rs,t[last].rs,tar-now);
} int main()
{
n=read();m=read();
for(int i=;i<=n;++i){
a[i]=read();b[i]=a[i];
}
sort(a+,a+n+);
tot=unique(a+,a+n+)-a-;
build(,tot,rt[]);
for(int i=;i<=n;++i)
rk[i]=lower_bound(a+,a+tot+,b[i])-a;
for(int i=;i<=n;i++)
update(,tot,rt[i],rt[i-],rk[i]);
int l,r,k;
for(int i=;i<=m;++i){
l=read();r=read();k=read();
printf("%d\n",quary(,tot,rt[r],rt[l-],k));
}
return ;
}
最新文章
- CSS 3学习——transition 过渡
- Mysql中eft join、right join、inner join的区别
- (int)、int.Parse()、int.TryParse()和Convert.ToInt32()的区别
- 利用DIV+CSS制作网页过程中常用的基本概念及标签使
- Xcode中Info.plist文件各个键的作用说明【搜藏】
- JavaScript中childNodes、children、nodeValue、nodeType、parentNode、nextSibling详细讲解
- 2015.4.16-C#中ref和out的区别
- DBCP数据源的使用
- curl 测试web站点的响应时间
- C++ Primer 笔记 第一章
- (转载)Sybase:bcp命令参考
- Healwire Online Pharmacy 3.0 Cross Site Request Forgery / Cross Site Scripting
- Python Django对接企业微信第三方服务回调验证的一些坑
- Caffe+CUDA8.0+CuDNNv5.1+OpenCV3.1+Ubuntu14.04 配置参考文献 以及 常见编译问题总结
- mezzanine的page_menu tag(二)
- 【读书笔记】iOS-iOS流媒体
- ThinkingInJava 学习 之 0000002 操作符
- VC/MFC程序开启关闭和打开自己或其他软件,更改窗口类
- umask命令详解
- Linux系统中提示/usr/bin/ld: cannot find -lxxx错误的通用解决方法