主席树(区间第k小的数)

Posted zxz666

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了主席树(区间第k小的数)相关的知识,希望对你有一定的参考价值。

题目链接: https://www.luogu.org/problem/P3834

首先要离散化,然后主席树模板。

 1 #include<cstdio>
 2 #include<cstring>
 3 #include<algorithm>
 4 #define mid (l+r)/2
 5 using namespace std;
 6 
 7 const int N = 200010;
 8 int n, q, m, cnt = 0;
 9 int a[N], b[N], T[N];
10 int sum[N<<5], L[N<<5], R[N<<5];
11 
12 inline int build(int l, int r)
13 
14     int rt = ++ cnt;
15     sum[rt] = 0;
16     if (l < r)
17         L[rt] = build(l, mid);
18         R[rt] = build(mid+1, r);
19     
20     return rt;
21 
22 
23 inline int update(int pre, int l, int r, int x)
24 
25     int rt = ++ cnt;
26     L[rt] = L[pre]; R[rt] = R[pre]; sum[rt] = sum[pre]+1;
27     if (l < r)
28         if (x <= mid) L[rt] = update(L[pre], l, mid, x);
29         else R[rt] = update(R[pre], mid+1, r, x);
30     
31     return rt;
32 
33 
34 inline int query(int u, int v, int l, int r, int k)
35 
36     if (l >= r) return l;
37     int x = sum[L[v]] - sum[L[u]];
38     if (x >= k) return query(L[u], L[v], l, mid, k);
39     else return query(R[u], R[v], mid+1, r, k-x);
40 
41 
42 int main()
43 
44     scanf("%d%d", &n, &q);
45     for (int i = 1; i <= n; i ++)
46         scanf("%d", &a[i]);
47         b[i] = a[i];
48     
49     sort(b+1, b+1+n);
50     m = unique(b+1, b+1+n)-b-1;
51     T[0] = build(1, m);
52     for (int i = 1; i <= n; i ++)
53         int t = lower_bound(b+1, b+1+m, a[i])-b;
54         T[i] = update(T[i-1], 1, m, t);
55     
56     while (q --)
57         int x, y, z;
58         scanf("%d%d%d", &x, &y, &z);
59         int t = query(T[x-1], T[y], 1, m, z);
60         printf("%d\n", b[t]);
61     
62     return 0;
63 

 

以上是关于主席树(区间第k小的数)的主要内容,如果未能解决你的问题,请参考以下文章

代码源 Div1 - 108#464. 数数(主席树,区间比k小的数的个数)HDU4417

代码源 Div1 - 108#464. 数数(主席树,区间比k小的数的个数)HDU4417

主席树铺垫——总区间第k小

HDU - 5919主席树区间不同数+区间第K大

主席树模板

[poj 2104]主席树+静态区间第k大