其他分享
首页 > 其他分享> > 255. 第K小数(主席树)

255. 第K小数(主席树)

作者:互联网

给定长度为 N 的整数序列 A,下标为 1∼N。

现在要执行 M 次操作,其中第 i 次操作为给出三个整数 li,ri,ki,求 A[li],A[li+1],…,A[ri] (即 A 的下标区间 [li,ri])中第 ki 小的数是多少。

输入格式
第一行包含两个整数 N 和 M。

第二行包含 N 个整数,表示整数序列 A。

接下来 M 行,每行包含三个整数 li,ri,ki,用以描述第 i 次操作。

输出格式
对于每次操作输出一个结果,表示在该次操作中,第 k 小的数的数值。

每个结果占一行。

数据范围
N≤105,M≤104,|A[i]|≤109

输入样例:
7 3
1 5 2 6 3 7 4
2 5 3
4 4 1
1 7 3
输出样例:
5
6
3
#include <iostream>
#include <cstring>
#include <algorithm>
#include <vector>
using namespace std;
const int N = 1e5+10, M = 1e4+10;
int w[N], root[N], idx;
vector<int> pos;
int n, m;
struct Node
{
    int l, r, cnt;
}t[N * 4 + 17 * N];
void build(int &u, int L, int R)
{
    u = ++ idx;
    if(L == R) return;
    int mid = L + R >> 1;
    build(t[u].l, L, mid); build(t[u].r, mid + 1, R);
}
int getp(int x)
{
    return lower_bound(pos.begin(), pos.end(), x) - pos.begin();
}
void insert(int p, int &q, int L, int R, int k)
{
    q = ++ idx; t[q] = t[p]; t[q].cnt ++;
    if(L == R) return;
    int mid = L + R >>1;
    if(k <= mid) insert(t[p].l, t[q].l, L, mid, k);
    if(k > mid) insert(t[p].r, t[q].r, mid + 1, R, k);
}
int query(int p, int q, int k, int L, int R)
{
    if(L == R) return L;
    int mid = L + R >> 1;
    int cnt = t[t[q].l].cnt - t[t[p].l].cnt;
    if(cnt >= k) return query(t[p].l, t[q].l, k, L, mid);
    if(cnt < k) return query(t[p].r, t[q].r, k - cnt, mid + 1, R);
    
}
int main()
{
    scanf("%d%d", &n, &m);
    for(int i = 1; i <= n; ++ i)
    {
        scanf("%d", &w[i]);
        pos.push_back(w[i]);
    }
    sort(pos.begin(), pos.end());
    pos.erase(unique(pos.begin(), pos.end()), pos.end());
    build(root[0], 0, pos.size() - 1);
    for(int i = 1; i <= n; ++ i) insert(root[i - 1], root[i], 0, pos.size() - 1, getp(w[i]));
    while(m -- )
    {
        int l, r, k; scanf("%d%d%d", &l, &r, &k);
        printf("%d\n", pos[query(root[l - 1], root[r], k, 0, pos.size() - 1)]);
    }
}

标签:cnt,return,int,mid,pos,li,255,主席,小数
来源: https://blog.csdn.net/m0_46656833/article/details/116563123