题目传送门

可持久化线段树1(主席树)

题目背景

这是个非常经典的主席树入门题——静态区间第K小

数据已经过加强,请使用主席树。同时请注意常数优化

题目描述

如题,给定N个正整数构成的序列,将对于指定的闭区间查询其区间内的第K小值。

输入输出格式

输入格式:

第一行包含两个正整数N、M,分别表示序列的长度和查询的个数。

第二行包含N个正整数,表示这个序列各项的数字。

接下来M行每行包含三个整数 $l,r,k$ , 表示查询区间 $[l,r]$ 内的第k小值。

输出格式:

输出包含k行,每行1个正整数,依次表示每一次查询的结果

输入输出样例

输入样例#1:

5 5
25957 6405 15770 26287 26465
2 2 1
3 4 1
4 5 1
1 2 2
4 4 1
输出样例#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 ;
}

洛谷P3834 [模板]可持久化线段树1(主席树) [主席树]的更多相关文章

  1. 【洛谷 P3834】 可持久化线段树1(主席树)

    题目链接 主席树=可持久化权值线段树. 如果你不会可持久化线段树,请右转 如果你不会权值线段树,请自行脑补,就是线段树维护值域里有多少个数出现. 可持久化线段树是支持查询历史版本的. 我们对每个数都进 ...

  2. 洛谷.3834.[模板]可持久化线段树(主席树 静态区间第k小)

    题目链接 //离散化后范围1~cnt不要错 #include<cstdio> #include<cctype> #include<algorithm> //#def ...

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

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

  4. 洛谷P3373 [模板]线段树 2(区间增减.乘 区间求和)

    To 洛谷.3373 [模板]线段树2 题目描述 如题,已知一个数列,你需要进行下面两种操作: 1.将某区间每一个数加上x 2.将某区间每一个数乘上x 3.求出某区间每一个数的和 输入输出格式 输入格 ...

  5. 【洛谷P3834】(模板)可持久化线段树 1(主席树)

    [模板]可持久化线段树 1(主席树) https://www.luogu.org/problemnew/show/P3834 主席树支持历史查询,空间复杂度为O(nlogn),需要动态开点 本题用一个 ...

  6. 洛谷P3834【模板】可持久化线段树 1(主席树)

    题目背景 这是个非常经典的主席树入门题--静态区间第K小 数据已经过加强,请使用主席树.同时请注意常数优化 题目描述 如题,给定N个正整数构成的序列,将对于指定的闭区间查询其区间内的第K小值. 输入输 ...

  7. HDU 2665.Kth number-可持久化线段树(无修改区间第K小)模板 (POJ 2104.K-th Number 、洛谷 P3834 【模板】可持久化线段树 1(主席树)只是输入格式不一样,其他几乎都一样的)

    Kth number Time Limit: 15000/5000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total ...

  8. LCT总结——概念篇+洛谷P3690[模板]Link Cut Tree(动态树)(LCT,Splay)

    为了优化体验(其实是强迫症),蒟蒻把总结拆成了两篇,方便不同学习阶段的Dalao们切换. LCT总结--应用篇戳这里 概念.性质简述 首先介绍一下链剖分的概念(感谢laofu的讲课) 链剖分,是指一类 ...

  9. 洛谷P4344 脑洞治疗仪 [SHOI2015] 线段树+二分答案/分块

    !!!一道巨恶心的数据结构题,做完当场爆炸:) 首先,如果你用位运算的时候不小心<<打成>>了,你就可以像我一样陷入疯狂的死循环改半个小时 然后,如果你改出来之后忘记把陷入死循 ...

随机推荐

  1. MSBuild问题积累

    我想要当属性ConfigurationType定义为StaticLibrary时,将其重新定义为StaticLibrary,按照以下来做,实现不了. <ConfigurationType> ...

  2. Eclipse中 如何实现 多行同时编辑

    在编辑的时候按下 SHIFT + ALT +A 之后  鼠标变为 + 号   选择要同时编辑几行  即可编辑(现在eclipse好像只能是编辑一块地方  不能像vs那样 任何地方可以同时编辑  这点很 ...

  3. CodeBlocks的常用快捷键

    CodeBlocks常用操作快捷键 编辑部分: Ctrl + A:全选 Ctrl + C:复制 Ctrl + X: 剪切 Ctrl + V:粘贴 Ctrl + Z:撤销 Ctrl + S:保存 Ctr ...

  4. parseInt

    本文地址:http://www.cnblogs.com/veinyin/p/7647863.html 先来个简单的 console.log(parseFloat("8")); 嗯, ...

  5. 【leetcode 简单】第三十一题 买卖股票的最佳时机

    给定一个数组,它的第 i 个元素是一支给定股票第 i 天的价格. 如果你最多只允许完成一笔交易(即买入和卖出一支股票),设计一个算法来计算你所能获取的最大利润. 注意你不能在买入股票前卖出股票. 示例 ...

  6. jQuery.pin.js笔记

    jQuery.pin.js是一个把元素钉在页面上某个位置的插件,它能够将某个元素一直挂在一个固定的位置而不论滚动条是否滚动. 特点: 1. 可以钉住一个元素,主要作用就是滚动超出的时候不会隐藏而是一直 ...

  7. 绿色的宠物店cms后台管理系统模板——后台

    链接:http://pan.baidu.com/s/1c7qmsA 密码:2es8

  8. 大图片上传(ImageIO,注意有的图片不能上传时因为他是tiff格式)

    一下是必要的: 1.enctype="multipart/form-data" 2. //不要使用myeclipse自动生成的get.set方法(struts2中的用法) publ ...

  9. Spring提供的iBatis的SqlMap配置

    1.    applicationContext.xml <!-- Spring提供的iBatis的SqlMap配置--> <bean id="sqlMapClient&q ...

  10. python并发编程之Queue线程、进程、协程通信(五)

    单线程.多线程之间.进程之间.协程之间很多时候需要协同完成工作,这个时候它们需要进行通讯.或者说为了解耦,普遍采用Queue,生产消费模式. 系列文章 python并发编程之threading线程(一 ...