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

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

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

主要思想:

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

所以空间就是 nlognn\log n 的。

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

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

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

code:code:

mark
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
#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_\_EOF\_