传送门:https://www.luogu.org/problemnew/show/P3834

题目描述

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

分析

很多人都说是用主席树来维护,一开始蒟蒻chh做这道题的时候还以为是分块乱搞,但是发现常数非常的大,就转用主席树了。这道题我们维护的主席树是一个权值线段树,意思就是维护一个桶,那么我们手动模拟主席树的建树过程,可以发现如果我们要查询区间\([l,r]\),那么发现编号为l-1的主席树和编号为r的主席树上对应的链上的差值就是我们区间上有多少个数是这样的。参照二叉搜索树的查找第k大的方法,我们就只需要每一次比较两个链上数值的差值和我们需要k大小,选择路径查询就可以做出来了。
注一句:这道题目因为维护的是权值线段树,那么题目中给出的1e9实在是太大了,那么我们就离散化一下,还是非常简单的。

ac代码

#include <bits/stdc++.h>
#define ll long long
#define ms(a, b) memset(a, b, sizeof(a))
#define inf 0x3f3f3f3f
#define N 200005
using namespace std;
template <typename T>
inline void read(T &x) {
    x = 0; T fl = 1;
    char ch = 0;
    while (ch < '0' || ch > '9') {
        if (ch == '-') fl = -1;
        ch = getchar();
    }
    while (ch >= '0' && ch <= '9') {
        x = (x << 1) + (x << 3) + (ch ^ 48);
        ch = getchar();
    }
    x *= fl;
}
struct Last_segment_tree {
    #define mid ((l + r)>> 1)
    int tot;
    int rt[N];
    struct node {
        int lc, rc, s;
        node() {
            lc = rc = s = 0;
        }
    }tr[N * 35];
    Last_segment_tree() {
        tot = 0;
    }
    void build(int &nod, int l, int r) {
        nod = ++ tot;
        if (l == r) return;
        build(tr[nod].lc, l, mid);
        build(tr[nod].rc, mid + 1, r);
    }
    void update(int &nod, int pre, int l, int r, int k) {
        nod = ++ tot;
        tr[nod] = tr[pre]; tr[nod].s ++;
        if (l == r) return;
        if (k <= mid) update(tr[nod].lc, tr[pre].lc, l, mid, k);
        else update(tr[nod].rc, tr[pre].rc, mid + 1, r, k);
    }
    int query(int x, int y, int l, int r, int k) {
        if (l == r) return l;
        int a = tr[tr[y].lc].s - tr[tr[x].lc].s;
        if (a >= k) return query(tr[x].lc, tr[y].lc, l, mid, k);
        else return query(tr[x].rc, tr[y].rc, mid + 1, r, k - a);
    }
    int kth(int l, int r, int ql, int qr, int k) {
        return query(rt[ql - 1], rt[qr], l, r, k);
    }
}lstr;
int n, m;
int a[N], disc[N];
int main() {
    read(n); read(m);
    for (int i = 1; i <= n; i ++) {
        read(a[i]);
        disc[i] = a[i];
    }
    sort(disc + 1, disc + 1 + n);
    int d = unique(disc + 1, disc + 1 + n) - disc - 1;
    lstr.build(lstr.rt[0], 1, d);
    for (int i = 1; i <= n; i ++) {
        a[i] = lower_bound(disc + 1, disc + 1 + d, a[i]) - disc;
        lstr.update(lstr.rt[i], lstr.rt[i - 1], 1 , d, a[i]);
    }
    while (m --) {
        int x, y, z;
        read(x); read(y); read(z);
        int p = lstr.kth(1, d, x, y, z);
        printf("%d\n", disc[p]);
    }
    return 0;
}

[luogu3834]静态区间第k小【主席树】的更多相关文章

  1. Dynamic Rankings || 动态/静态区间第k小(主席树)

    JYF大佬说,一星期要写很多篇博客才会有人看 但是我做题没有那么快啊QwQ Part1 写在前面 区间第K小问题一直是主席树经典题=w=今天的重点是动态区间第K小问题.静态问题要求查询一个区间内的第k ...

  2. POJ2104 K-th Number —— 静态区间第k小

    题目链接:http://poj.org/problem?id=2104 K-th Number Time Limit: 20000MS   Memory Limit: 65536K Total Sub ...

  3. 静态区间第K小(整体二分、主席树)

    题目链接 题解 主席树入门题 但是这里给出整体二分解法 整体二分顾名思义是把所有操作放在一起二分 想想,如果求\([1-n]\)的第\(k\)小怎么二分求得? 我们可以二分答案\(k\), \(O(n ...

  4. 主席树(可持久化线段树)静态区间第K小

    传送门主席树 #include <bits/stdc++.h> #define int long long using namespace std; const int maxn=2e5+ ...

  5. POJ2104&&HDU2665(静态区间第K小)

    题目大意 给定一个有N个数字的序列,然后又m个查询,形式如下: l r k 要求你返回区间[l,r]第k小的数是哪个 题解 终于弄懂主席树是个啥东西了,O(∩_∩)O~~,这题正是主席树的裸题,主席树 ...

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

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

  7. HDU2665 求区间第K大 主席树

    题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=2665 代码: //#include<bits/stdc++.h> #include< ...

  8. 静态区间第k小 - 整体二分

    蒟蒻终于学会整体二分啦! 思路 实现 丑陋无比的代码 #include <bits/stdc++.h> using namespace std; const int N = 200005; ...

  9. POJ-2104-K-th Number(区间第K大+主席树模板题)

    Description You are working for Macrohard company in data structures department. After failing your ...

随机推荐

  1. 使用IE浏览提示:该页面无法显示

    问题描述: 我们有一个外部招聘的网站,DBA反馈新版上线过后首页集成的登录部分页面无法打开,一直显示“该页面无法显示”! 问题排查: 1.因为我本身也不是负责这一块的业务,刚开始以为是网站本身程序的问 ...

  2. NOIP2018 模拟赛(二十二)雅礼NOI

    Preface 这次的题目都是NOI+的题,所以大家的分数都有点惨烈. 依靠T1大力骗分水到Rank2 所以想看正解的话看这里吧 A. 「雅礼NOI2018模拟赛(一) Day1」树 看一眼题目感觉十 ...

  3. 2018 Multi-University Training Contest 1 部分简单题解析

    Preface ACM系列赛第一站,没有进前200还是很伤的. 主要是T2当时没写出来就GG了,后来看了下其实不是很难. 题目按照比赛时我们A的顺序讲,其实我都是被陈潇然大佬和ZWC带飞的. T1 M ...

  4. loj6062 pair

    直接套用霍尔定理. 由于A有多个选择,考虑维护B是否合法. 首先B数组的顺序显然是没有用的,可以直接排序. 然后每个A就都变成了向一个后缀连边. 对于B,原本需要check每一个集合是否满足|u|&l ...

  5. C#_面试

    class Program { static void Main(string[] args) { , , , , }; var arry = ConvertSum(arr); , , , , , } ...

  6. grep精确匹配搜索某个单词的用法 (附: grep高效用法小结))

    grep(global search regular expression(RE) and print out the line,全面搜索正则表达式并把行打印出来)是一种强大的文本搜索工具,它能使用正 ...

  7. UML图中类之间的关系:依赖,泛化,关联,聚合,组合,实现(转)

    UML图中类之间的关系:依赖,泛化,关联,聚合,组合,实现   类与类图 1) 类(Class)封装了数据和行为,是面向对象的重要组成部分,它是具有相同属性.操作.关系的对象集合的总称. 2) 在系统 ...

  8. Visual Studio 2017 社区版的安装与组件修改(C++)

    0. 环境描述 需求:用VS2017做C++简易开发. 操作系统:Windows 8.1. 1. 下载 MSDN下载VS2017社区版. https://msdn.itellyou.cn/ 下载后: ...

  9. Java基础知识中的注意事项

    设置Java的相关路径,举例: JAVA_HOME  -->   C:\Program Files\Java\jdk1.8.0_191 Path --->   C:\Program Fil ...

  10. Protobuf一例

    Developer Guide  |  Protocol Buffers  |  Google Developershttps://developers.google.com/protocol-buf ...