重码数据结构主席树(可持久化线段树)

Posted chriskkk

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了重码数据结构主席树(可持久化线段树)相关的知识,希望对你有一定的参考价值。

例题:https://www.luogu.org/problemnew/show/P3834

主席树用于查询每个历史版本。

这个题代码如下

#include<bits/stdc++.h>
#define ll long long
using namespace std;
const int maxn=200001;
int n,m,tot;
ll a[maxn],b[maxn];
struct stree
    int lc,rc;
    ll sum;
    #define lc(x) tree[x].lc
    #define rc(x) tree[x].rc
    #define s(x) tree[x].sum
tree[maxn<<5];
int root[maxn];
inline int build(int l,int r)
    int p=++tot;
    if(l==r) return p;
    int mid=(l+r)>>1;
    lc(p)=build(l,mid);
    rc(p)=build(mid+1,r);
    return p;

inline int insert(int now,int l,int r,int x)
    int p=++tot;
    tree[p]=tree[now];
    s(p)++;
    if(l==r)return p;
    int mid=(l+r)>>1;
    if(x<=mid) lc(p)=insert(lc(now),l,mid,x);
    else rc(p)=insert(rc(now),mid+1,r,x);
    return p;

inline int ask(int p,int q,int l,int r,int k)//在p,q两个时间节点上,在[l,r]的范围内的数的个数
    if(l==r) return l;
    int mid=(l+r)>>1;
    int lcnt=s(lc(q))-s(lc(p));
    if(k<=lcnt) return ask(lc(p),lc(q),l,mid,k);//在两个节点的左儿子中查找。
    else return ask(rc(p),rc(q),mid+1,r,k-lcnt);

int main()

    scanf("%d%d",&n,&m);
    for(int i=1;i<=n;i++)
        scanf("%lld",&a[i]),b[i]=a[i];
    sort(b+1,b+n+1);
    int cnt=unique(b+1,b+n+1)-b-1;
    root[0]=build(1,cnt);
    for(int i=1;i<=n;i++)
        a[i]=lower_bound(b+1,b+cnt+1,a[i])-b;
        root[i]=insert(root[i-1],1,cnt,a[i]);
    
    while(m--)
        int l,r,k;
        scanf("%d%d%d",&l,&r,&k);
        printf("%lld\n",b[ask(root[l-1],root[r],1,cnt,k)]);
    
    system("pause");
    return 0;

 

 

例题2:https://www.luogu.org/problemnew/show/P3919

主席树裸题,直接就是模板。

代码如下:

#include<bits/stdc++.h>
using namespace std;
const int maxn=1e6+10;
int n,m,tot;
int root[maxn],a[maxn];
struct stree
    int lc,rc,val;
    #define lc(x) tree[x].lc
    #define rc(x) tree[x].rc
    #define val(x) tree[x].val 
tree[maxn<<5];
inline int build(int l,int r)
    int p=++tot;
    if(l==r)val(p)=a[l];return p;
    int mid=(l+r)>>1;
    lc(p)=build(l,mid);
    rc(p)=build(mid+1,r);
    return p;

inline int insert(int now,int l,int r,int x,int dat)
    int p=++tot;
    tree[p]=tree[now];
    if(l==r)val(p)=dat;return p;
    int mid=(l+r)>>1;
    if(x<=mid) lc(p)=insert(lc(now),l,mid,x,dat);
    else rc(p)=insert(rc(now),mid+1,r,x,dat);
    return p;

inline int ask(int p,int l,int r,int x)
    if(l==r) return val(p);
    int mid=(l+r)>>1;
    if(x<=mid) return ask(lc(p),l,mid,x);
    else return ask(rc(p),mid+1,r,x);

int main()

    scanf("%d%d",&n,&m);
    for(int i=1;i<=n;i++) scanf("%d",&a[i]);
    root[0]=build(1,n);
    for(int i=1;i<=m;i++)
        int v,op,x,c;
        scanf("%d%d%d",&v,&op,&x);
        if(op==1) scanf("%d",&c),root[i]=insert(root[v],1,n,x,c);
        else printf("%d\n",ask(root[v],1,n,x)),root[i]=root[v];
    
    // system("pause");
    return 0;

 

以上是关于重码数据结构主席树(可持久化线段树)的主要内容,如果未能解决你的问题,请参考以下文章

P3834 模板可持久化线段树 1(主席树)

可持久化线段树--主席树

可持久化线段树/主席树(静态)

LUOGU P3834 模板可持久化线段树 1(主席树)

可持续化线段树(主席树)

可持久化线段树(主席树)