题目描述

给定n个整数构成的序列,将对于指定的闭区间查询其区间内的第k小值。

输入输出格式

输入格式

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

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

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

输出格式

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

输入输出样例

输入样例

5 5
5957 6405 15770 26287 26465
2 2 1
3 4 1
4 5 1
1 2 2
4 4 1

输出样例

6405
15770
26287
25957
26287

题解

直接康代码吧,个人觉得注释还是蛮详细的

#include <bits/stdc++.h>
using namespace std;
const int N=2e5+2;
int tot,n,m,l,r,k;//tot 所有节点的数量
struct node{
int lc,rc;//表编号为i的节点的左/右儿子的编号
int cnt;//表编号为i的节点所代表的区间内数字出现的次数
}tr[N<<5];
int a[N],b[N];//a[i]为原数组 b[i]为排序后数组
int rt[N];//插入i个点后的树的根节点编号
int build(int l,int r) {//建一个空树(所有cnt都为0)
int p=++tot; //p为当前节点编号
if(l==r) return p;
int mid=(l+r)>>1;
tr[p].lc=build(l,mid);
tr[p].rc=build(mid+1,r);
return p; //返回当前节点的编号
}
int update(int pre,int l,int r,int x) { //pre为旧树该位置节点下标
// x为 修改位置下标
int p=++tot; //新建节点
tr[p].lc=tr[pre].lc;
tr[p].rc=tr[pre].rc;
tr[p].cnt=tr[pre].cnt+1;
//copy 原节点信息
if(l==r) return p;//到达底层,回溯
int mid=(l+r)>>1;
if(x<=mid) tr[p].lc=update(tr[pre].lc,l,mid,x);
//x出现在左子树 因此右子树保持与旧树相同 修改左子树
else tr[p].rc=update(tr[pre].rc,mid+1,r,x);
//否则前往右子树
return p;//返回线段树中的编号
}
int ask(int u,int v,int l,int r,int k) {
//在u,v两个节点上,数域为[l,r],求第k小数
if(l==r) return b[l]; //找到第k小 l/r是节点编号 所以答案是b[l/r]
int mid=(l+r)>>1;
int p=tr[tr[v].lc].cnt-tr[tr[u].lc].cnt;
//则p= (1~r)树的左节点数字出现的次数 - (1~(l-1))树的左节点数字出现的次数
//即p等于([l,r])树左儿子数字出现的次数
if(p>=k) return ask(tr[u].lc,tr[v].lc,l,mid,k);
//第k小的数字在左子树处
else return ask(tr[u].rc,tr[v].rc,mid+1,r,k-p);
//否则去右子树处找第k-p小的数字
}
int main(){
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++) {
scanf("%d",&a[i]);
b[i]=a[i];
}
sort(b+1,b+1+n);
int size=unique(b+1,b+1+n)-b-1;
//size为线段树维护的数组的大小,即b数组中不重复的数字的个数
rt[0]=build(1,size); //初始化 建立一颗空树 并把该树的根节点的编号赋值给rt[0]
for(int i=1;i<=n;i++){
int x=lower_bound(b+1,b+1+size,a[i])-b;
//在b的 [1,size+1)--->[1,size] 中二分查找第一个大于等于a[i]的b[x]
rt[i]=update(rt[i-1],1,size,x);
//更新a[i]带来的影响
//并将新树的根节点的编号赋值给rt[i]
}
while(m--){
scanf("%d%d%d",&l,&r,&k);
int ans=ask(rt[l-1],rt[r],1,size,k);
printf("%d\n",ans);
}
return 0;
}

完结撒花❀

「洛谷 P3834」「模板」可持久化线段树 题解报告的更多相关文章

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

    题目 传送门:QWQ 分析 主席树的模板,囤着 代码 #include <bits/stdc++.h> using namespace std; ; ], rs[N*], root[N*] ...

  2. 【整体二分】【P3834】 【模板】可持久化线段树 1(主席树)

    Description 给定一个长度为 \(n\) 的序列, \(m\) 次操作静态查询区间第 \(k\) 大 Input 第一行是 \(n,m\) 下一行描述这个序列 下面 \(m\) 行描述操作 ...

  3. 【莫队】【P3834】 【模板】可持久化线段树 1(主席树)

    大家好,我是个毒瘤,我非常喜欢暴力数据结构,于是我就用莫队+分块过了这个题 Solution 发现这个题静态查询资瓷离线,于是考虑莫队. 在这里简单介绍一下莫队: 将所有询问离线后,对原序列分块.按照 ...

  4. 题解 P3834 【【模板】可持久化线段树 1(主席树)】

    可持久化线段树的前置知识是权值线段树,但是你不学也没有太大的关系因为思想不是很难理解. 可持久化线段树支持历史记录查询,这是它赖以解题的方法. 在本题中思路是建立n颗线段树,然后对于每次询问,考虑其中 ...

  5. LuoguP3834 【模板】可持久化线段树 1(主席树)|| 离散化

    题目:[模板]可持久化线段树 1(主席树) 不知道说啥. #include<cstdio> #include<cstring> #include<iostream> ...

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

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

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

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

  8. luogu P3834 【模板】可持久化线段树 1(主席树) 查询区间 [l, r] 内的第 k 小/大值

    ————————————————版权声明:本文为CSDN博主「ModestCoder_」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明.原文链接:https:// ...

  9. 洛谷P3178 [HAOI2015]树上操作(dfs序+线段树)

    P3178 [HAOI2015]树上操作 题目链接:https://www.luogu.org/problemnew/show/P3178 题目描述 有一棵点数为 N 的树,以点 1 为根,且树点有边 ...

随机推荐

  1. CSS3中有哪些新特性

    新增各种CSS选择器 (: not(.input):所有 class 不是"input"的节点) 圆角 (border-radius:8px) 多列布局 (multi-column ...

  2. 申请百度翻译API

    申请百度翻译API 0x00 前期准备 百度账号 md5的相关知识 0x01 进入百度开放平台,登录你的百度账号 找到 产品服务 -> 通用翻译API 0x02 点击下面的立即使用按钮,注册成为 ...

  3. 线性表(python实现)

    线性表 1 定义 线性表是由 \(n(n>=0)\)个数据元素(节点)\(a1.a2.a3.-.an\) 成的有限序列.该序列中的所有节点都具有相同的数据类型.其中,数据元素的个数 \(n\) ...

  4. asp.net core启动源码以及监听,到处理请求响应的过程

    摘要 asp.net core发布至今已经将近6年了,很多人对于这一块还是有些陌生,或者说没接触过:接触过的,对于asp.net core整个启动过程,监听过程,以及请求过程,响应过程也是一知半解,可 ...

  5. 《手把手教你》系列基础篇(八十五)-java+ selenium自动化测试-框架设计基础-TestNG自定义日志-下篇(详解教程)

    1.简介 TestNG为日志记录和报告提供的不同选项.现在,宏哥讲解分享如何开始使用它们.首先,我们将编写一个示例程序,在该程序中我们将使用 ITestListener方法进行日志记录. 2.Test ...

  6. selenium打开指定Chrome账号

    selenium打开指定Chrome账号 获取User Data路径 打开目标Chrome,在搜索栏输入chrome://version,找到"个人资料路径". 这里获取到的路径为 ...

  7. Python图像处理丨OpenCV+Numpy库读取与修改像素

    摘要:本篇文章主要讲解 OpenCV+Numpy 图像处理基础知识,包括读取像素和修改像素. 本文分享自华为云社区<[Python图像处理] 二.OpenCV+Numpy库读取与修改像素> ...

  8. C# 滑动验证码|拼图验证|SlideCaptcha

    使用背景: 关于滑动验证码的使用场所还是非常多的,如: 调取短信接口之前,和 注册请求之前 或者 频繁会调用的接口 都需要加这个拼图验证.这里先上一下效果图吧(心中无码,自然高清). 话不多说,开撸! ...

  9. 【第三课】常用的Linux命令(学习笔记)

    4月8日 学习笔记打卡

  10. synchronized锁及其锁升级

    点赞再看,养成习惯,微信搜索「小大白日志」关注这个搬砖人. 文章不定期同步公众号,还有各种一线大厂面试原题.我的学习系列笔记. 多线程加锁有两种方式 利用Sychronized关键字 利用Lock接口 ...