一个很好的主席树总结

码一下:https://blog.csdn.net/weixin_44584560/article/details/87115412

#include <bits/stdc++.h>
using namespace std;
const int N = 100000 + 5;

int a[N] ,b[N] ,rt[N*20] ,ls[N*20] ,rs[N*20] ,sum[N*20];
int   n ,k ,tot ,sz ,ql ,qr ,x ,q ,T;

void Build( int& o ,int l ,int r){      //i=0时间的前缀状态 :0树 
    o = ++tot;
    sum[o] = 0;
    if( l==r ) return;
    int m = (l+r) >>1;
    Build( ls[o] , l ,m);
    Build( rs[o] ,m+1 ,r);
}

void update( int &o ,int l ,int r ,int last ,int p ){
    o = ++tot;             //伴随着rt引用o的更新 ,rt随之更新  
    ls[o] = ls[last];
    rs[o] = rs[last];
    sum[o] = sum[last] + 1;
    if( l == r) return ;
    int m = ( l + r )>>1;
    if( p <= m) update( ls[o] ,l ,m ,ls[last] ,p);
    if( p > m ) update( rs[o] ,m+1,r ,rs[last] ,p);
}

int query( int ss ,int tt ,int l ,int r ,int k ){
     if( l == r) return l;
     int m = ( l + r )>>1;
     int cnt = sum[ ls[tt] ] - sum[ ls[ss] ];
     if( k <= cnt ) return query( ls[ss] ,ls[tt] ,l ,m ,k);
     else return query( rs[ss] ,rs[tt] ,m+1 ,r ,k - cnt);
}

void work(){
    scanf("%d%d%d" ,&ql ,&qr ,&x );
    int ans = query( rt[ ql-1],rt[qr] ,1 ,sz ,x);
    printf( "%d\n" ,b[ans] );
}

int main( ){
    scanf("%d" ,&T);
    while( T-- ){
        scanf( "%d%d" ,&n ,&q );
        for( int i=1 ;i<=n ;i++ )
        scanf("%d" ,&a[i]) ,b[i] = a[i];
        
        sort( b+1 , b+n+1 );
        sz = unique( b+1 ,b+n+1 ) - ( b+1 );
        tot = 0;
        Build( rt[0] ,1 ,sz );
        //rt[i],插入时的时间i对应的结点编号 ,编号插入先序 
        for( int i=1 ;i<=n ;i++ )a[i] = lower_bound( b+1 ,b+1+sz ,a[i]) - b;
        for( int i=1 ;i<=n ;i++ )update( rt[i] ,1 ,sz ,rt[ i - 1] ,a[i]);
        while( q-- )work();
    }
    return 0;
}
posted @ 2019-07-29 20:46  易如鱼  阅读(212)  评论(0编辑  收藏  举报