可持久化线段树的前置知识是权值线段树,但是你不学也没有太大的关系因为思想不是很难理解。


可持久化线段树支持历史记录查询,这是它赖以解题的方法。

在本题中思路是建立n颗线段树,然后对于每次询问,考虑其中两颗:由于这n颗线段树的结构都一样,而且区间是可以加减的,所以我们将它们相减,再搜一下就可以得到第K大了。


我们每一颗线段树,维护的是元素在区间中出现的次数。

假如直接建n颗树,空间不能接受,因此我们可以考虑简化。可以发现,这些区间都是[1,i]的形式,因此左半部分事实上都是相同的,每一次建立新的树我们只用右半部分新建,重复利用左半部分即可,这样的话空间就可以接受了。


重要变量

int a[N],b[N],T[N];
int sum[N*DEPTH],L[N*DEPTH],R[N*DEPTH];

a数组是题目给出的数组。

b数组是经过排序去重之后的数组。

T数组储存每一颗线段树的根节点(和普通线段树不一样,可持久化线段树的建树和更新都有返回值)。

sum数组就是区间中数的个数。

L和R表示左儿子与右儿子。


建树

inline int build(int l,int r){
    int pos=tot++,mid=(l+r)>>1;
    if(l<r){
        build(l,mid),build(mid+1,r);
    }
    return pos;
}

首先先新建节点,然后递归建树,没什么好讲的。

注意函数末尾返回新建节点的编号(用于T数组的赋值)。


更新

inline int update(int l,int r,int v,int pre){
    int pos=tot++,mid=(l+r)>>1;
    L[pos]=L[pre],R[pos]=R[pre],sum[pos]=sum[pre]+1;
    if(l<r){
        if(v<=mid)L[pos]=update(l,mid,v,L[pre]);
        else R[pos]=update(mid+1,r,v,R[pre]);
    }
    return pos;
}

和普通线段树的更新相比,可持久化线段树的少了一个pos表示当前位置,多了一个pre表示之前的。

首先还是先新建一个节点(毕竟我们可持久化就是一颗颗树),然后新的这一个部分先连向之前那个节点的左右儿子。

这个时候,我们再根据更新的值去重新确定新的左右儿子。

然后最后返回节点编号(为了顶上的确定流程)。


查询

inline int query(int phase1,int phase2,int l,int r,int k){
    int t=sum[L[phase2]]-sum[L[phase1]],mid=(l+r)>>1;
    if(l<r){
        if(k<=t)return query(L[phase1],L[phase2],l,mid,k);
        return query(R[phase1],R[phase2],mid+1,r,k-t);
    }
}

查询操作比较好理解,有点像差分。

就是我们有两个版本的线段树,然后拿他们相减,一路往下找,最后拿到答案即可。


然后就写完了。

代码如下:

#include<iostream>
#include<cstdio>
#include<algorithm>
using namespace std;
namespace ElderTree{
    const qwq N=200200,DEPTH=23;
    inline qwq read(){
        qwq x=0,f=1;char c=getchar();
        for(;c<'0'||c>'9';c=getchar())if(c=='-')f=-1;
        for(;c>='0'&&c<='9';c=getchar())x=x*10+c-'0';
        return x*f;
    }
    qwq n,m,q,tot;
    qwq a[N],b[N],T[N];
    qwq sum[N*DEPTH],L[N*DEPTH],R[N*DEPTH];
    inline qwq build(qwq l,qwq r){
        qwq pos=tot++,mid=(l+r)>>1;
        if(l<r){
            build(l,mid),build(mid+1,r);
        }
        return pos;
    }
    inline qwq update(qwq l,qwq r,qwq v,qwq pre){
        qwq pos=tot++,mid=(l+r)>>1;
        L[pos]=L[pre],R[pos]=R[pre],sum[pos]=sum[pre]+1;
        if(l<r){
            if(v<=mid)L[pos]=update(l,mid,v,L[pre]);
            else R[pos]=update(mid+1,r,v,R[pre]);
        }
        return pos;
    }
    inline qwq query(qwq phase1,qwq phase2,qwq l,qwq r,qwq k){
        qwq t=sum[L[phase2]]-sum[L[phase1]],mid=(l+r)>>1;
        if(l<r){
            if(k<=t)return query(L[phase1],L[phase2],l,mid,k);
            return query(R[phase1],R[phase2],mid+1,r,k-t);
        }
    }
}
using namespace ElderTree;
qwq main(){
    n=read(),q=read();
    for(qwq i=1;i<=n;++i)a[i]=read(),b[i]=a[i];
    sort(b+1,b+n+1),m=unique(b+1,b+n+1)-1-b,T[0]=build(1,m);
    for(qwq i=1;i<=n;++i){
        qwq t=lower_bound(b+1,b+m+1,a[i])-b;
        T[i]=update(1,m,t,T[i-1]);
    }
    while(q--){
        qwq x,y,z;
        x=read(),y=read(),z=read();
        cout<<b[query(T[x-1],T[y],1,m,z)]<<endl;
    }
    return 0;
}

题解 P3834 【【模板】可持久化线段树 1(主席树)】的更多相关文章

  1. 洛谷P3834 [模板]可持久化线段树1(主席树) [主席树]

    题目传送门 可持久化线段树1(主席树) 题目背景 这是个非常经典的主席树入门题——静态区间第K小 数据已经过加强,请使用主席树.同时请注意常数优化 题目描述 如题,给定N个正整数构成的序列,将对于指定 ...

  2. P3919 【模板】可持久化数组 -初步探究主席树

    本篇blog主要是给自己(大家)看的. 感谢longlongzhu123奆佬(此人初二LCT)的指点,使本蒟蒻可以快速开始主席树入门. what is 主席树? $        $主席树这个名字只不 ...

  3. 归并树 划分树 可持久化线段树(主席树) 入门题 hdu 2665

    如果题目给出1e5的数据范围,,以前只会用n*log(n)的方法去想 今天学了一下两三种n*n*log(n)的数据结构 他们就是大名鼎鼎的 归并树 划分树 主席树,,,, 首先来说两个问题,,区间第k ...

  4. 【题解】BZOJ3489 A Hard RMQ problem(主席树套主席树)

    [题解]A simple RMQ problem 占坑,免得咕咕咕了,争取在2h内写出代码 upd:由于博主太菜而且硬是要用指针写两个主席树,所以延后2hQAQ upd:由于博主太菜而且太懒所以他决定 ...

  5. POJ 2104 K-th Number(分桶,线段树,主席树)

    一道比较经典的数据结构题.可以用多种方式来做. 一,分桶法(平方分解). 根据数字x的大小和区间内不大于x的数字数量cnt的单调性,可知第k大数kth对应的cnt应该满足cnt≥k, 且kth是满足条 ...

  6. 【BZOJ4771】七彩树(主席树)

    [BZOJ4771]七彩树(主席树) 题面 BZOJ 题解 如果没有深度限制,每次只询问子树内的颜色个数,除了树套树\(dfs\)序加前驱或者后继强行二维数点之外,还有这样一种做法: 把所有相同颜色的 ...

  7. 洛谷P3248 树 [HNOI2016] 主席树+倍增+分治

    正解:主席树+倍增+分治 解题报告: 传送门! 首先看到这题会想到之前考过的这题 但是那题其实简单一些,,,因为那题只要用个分治+预处理就好,只是有点儿思维难度而已 这题就不一样,因为它说了是按照原树 ...

  8. poj 2104 K-th Number 划分树,主席树讲解

    K-th Number Input The first line of the input file contains n --- the size of the array, and m --- t ...

  9. BZOJ_2588_Spoj 10628. Count on a tree_树剖+主席树

    BZOJ_2588_Spoj 10628. Count on a tree_树剖+主席树 题意: 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastan ...

  10. POJ 2761 Feed the dogs(平衡树or划分树or主席树)

    Description Wind loves pretty dogs very much, and she has n pet dogs. So Jiajia has to feed the dogs ...

随机推荐

  1. Cookie和Session有什么区别

    1. 由于HTTP协议是无状态的协议,所以服务端需要记录用户的状态时,就需要用某种机制来识别具体的用户,这个机制就是Session.   典型的场景比如购物车,当你点击下单按钮时,由于HTTP协议无状 ...

  2. python3 继承与组合

    什么叫继承? 所谓继承,就是class_A里面的功能从class_B中直接获取,从而节约了代码且使用方便. 什么叫组合? 除了继承,还有一种我们可以实现目的的方式,那就是组合,同样可以节约代码.只不过 ...

  3. python3三级菜单的访问,并按q退出

    #/usr/bin/env python#yehui'''作业三:多级菜单 三级菜单 可依次选择进入各子菜单 所需新知识点:列表.字典'''import readlineclass MultiLeve ...

  4. (56) 解决字段设为readonly无法保存

    问题描述:当一个字段设为readonly =True 后,在form表单,即使你用onchange方法改变了值但也不能保存到数据库当时.平时在这样的要求,form表单有些字段要展示给用户,但又要达到不 ...

  5. [luogu] P4040 [AHOI2014/JSOI2014]宅男计划(贪心)

    P4040 [AHOI2014/JSOI2014]宅男计划 题目背景 自从迷上了拼图,JYY就变成了个彻底的宅男.为了解决温饱问题,JYY不得不依靠叫外卖来维持生计. 题目描述 外卖店一共有N种食物, ...

  6. jenkins 安装遇到的坑

                        最后启用安全的时候遇到一点坑,直接写了一个用户上去,没有勾选权限,然后在登录就说没有 overright/等,然后需要修改配置文件conf.xml 在主目录下. ...

  7. SQL SERVER-集合操作

    内连接      INNER JOIN(等值连接):只显示两个表中联结字段相等的行.这个和用select查询多表是一样的效果,所以很少用到:外连接:LEFT JOIN :以左表为基础,显示左表中的所有 ...

  8. 求第K大的数字

    除了用最大堆(求最小的K个数)或最小堆(求最大的K个数) 可以用partition,然后直到返回index为k为止.参数可以是实际下标.然后返回index,就是partition的pivot的位置.

  9. POJ 2906 数学期望

    开始时直接设了一个状态,dp[i][j]为发现i种bug,j个系统有bug的期望天数.但很错误,没能转移下去.... 看了题解,设状态dp[i][j]为已发现i种bug,j个系统有bug,到完成目标状 ...

  10. 广播BroadcastReceiver(2)

    有序广播的优先级:        发送有序广播的方法有:    public void sendOrderedBroadcast(Intent intent,String receiverPermis ...