『学习笔记』可持久化线段树(主席树)

主席树的最基础的操作就是查询历史版本区间第 \(k\) 大,带修。

这个问题的基础解决思路:对于每次修改都建一棵权值线段树,显然空间开不下。

这时可持久化线段树的思路就应运而生了。

主要思想:

不难发现,每次修改只会有一条链上的值发生改变,所以我们不需要建出整棵新树,只需要把新建那条链上的点即可。

所以空间就是 \(n\log n\) 的。

我们还要在权值线段树上做个前缀和来方便查询,每次查询时,就是在权值线段树上二分的过程。

注意到我们要建的是权值线段树,所以要对输入的值做一个离散化。

不多说了,直接贴代码吧。

\(code:\)

#include <bits/stdc++.h>
#define uint unsigned int
#define ls(x) t[x].l
#define rs(x) t[x].r

using namespace std;

namespace IO{
    inline int read(){
        int x = 0, f = 1;
        char ch = getchar();
        while(!isdigit(ch)) {if(ch == '-') f = -1; ch = getchar();}
        while(isdigit(ch)) x = (x << 3) + (x << 1) + ch - '0', ch = getchar();
        return x * f;
    }

    inline char readc(){
        char ch = getchar();
        while(ch < 'a' || ch > 'z') ch = getchar();
        return ch;
    }

    template <typename T> inline void write(T x){
        if(x > 9) write(x / 10);
        putchar(x % 10 + '0');
    }
}
using namespace IO;

const int N = 2e5 + 10;
int n, m, cnt;
int a[N], b[N];
struct Tree{
    int rt, sum, l, r;
    Tree() {sum = 0;}
}t[N << 5];

inline int build(int l, int r){
    int rt = ++cnt;
    if(l == r) return rt;
    int mid = (l + r) >> 1;
    ls(rt) = build(l, mid);
    rs(rt) = build(mid + 1, r);
    return rt;
}

inline int update(int pre, int l, int r, int x){
    int rt = ++cnt;
    ls(rt) = ls(pre), rs(rt) = rs(pre), t[rt].sum = t[pre].sum + 1;
    if(l == r) return rt;
    int mid = (l + r) >> 1;
    if(x <= mid) ls(rt) = update(ls(pre), l, mid, x);
    else rs(rt) = update(rs(pre), mid + 1, r, x);
    return rt;
}

inline int query(int L, int R, int k, int l, int r){
    if(l == r) return l;
    int res = t[ls(R)].sum - t[ls(L)].sum;
    int mid = (l + r) >> 1;
    if(res >= k) return query(ls(L), ls(R), k, l, mid);
    else return query(rs(L), rs(R), k - res, mid + 1, r);
}

int main(){
    n = read(), m = read();
    for(int i = 1; i <= n; ++i) a[i] = read(), b[i] = a[i];
    sort(b + 1, b + 1 + n);
    int tot = unique(b + 1, b + 1 + n) - b - 1;
    for(int i = 1; i <= n; ++i)
        a[i] = lower_bound(b + 1, b + 1 + tot, a[i]) - b;
    t[0].rt = build(1, tot);
    for(int i = 1; i <= n; ++i) t[i].rt = update(t[i - 1].rt, 1, tot, a[i]);
    while(m--){
        int l = read(), r = read(), k = read();
        write(b[query(t[l - 1].rt, t[r].rt, k, 1, tot)]), puts("");
    }
    return 0;
}

\[\_EOF\_ \]

posted @ 2021-12-27 14:02  xixike  阅读(33)  评论(0编辑  收藏  举报