首页 > 其他 > 详细

权值线段树&&可持久化线段树&&主席树

时间:2019-10-01 17:31:20      阅读:157      评论:0      收藏:0      [点我收藏+]

权值线段树

顾名思义,就是以权值为下标建立的线段树。

现在让我们来考虑考虑上面那句话的产生的三个小问题:

1. 如果说权值作为下标了,那这颗线段树里存什么呢?
————— 这颗线段树中, 记录每个值出现的次数

2.权值很大怎么办?数组空间不够啊
————— 可以先离散化,再记录

3.那权值线段树到底是用来干嘛的呢?
————— 可以快速求出每个权值出现的次数(其实主要还是为了主席树做铺垫啦)

代码就不给了

可持久化线段树

普通的线段树单点修改操作与区间查询自然不是问题
可是
假如当前询问若干修改操作之前的区间呢???

仔细想想
.
.
.
最暴力的做法无疑是对于每个修改操作重开一个线段树,
可是...这样显然空间开不下
那我们能不能优化一下呢

我们看看对于一次单点修改,这颗线段树操作前和操作后有什么不同吧

技术分享图片

有点小丑,凑合着看
观察一下这两颗树,发现它们有区别的地方仅仅在于红色的方框
哎??? 这不是此次操作修改的目标元素到根的路径吗

既然只有这条路径变了,那我们就只复制这条路径好了,不用再复制整棵树了
所以空间就能大大的缩小了(log级)

代码 (洛谷模板)

#include<bits/stdc++.h>
using namespace std;
#define re register
#define ll long long
#define get getchar()
#define in inline
in int read()
{
    int x=1,t=0; char ch=get;
    while((ch<'0' || ch>'9') && ch!='-') ch=get;
    if(ch=='-') ch=get,x=-1;
    while(ch<='9' && ch>='0') t=t*10+ch-'0', ch=get;
    return t*x;
}
const int _=1e6+6;
int n,m,a[_],tot,root[_<<5],ls[_<<5],rs[_<<5],val[_<<5]; // ls == leftson,rs == rightson
in int build(int l,int r)
{
    int now=++tot;
    if(l==r)
    {
        ls[now]=rs[now]=0;
        val[now]=a[l];
        return now;
    }
    int mid=(l+r)>>1;
    ls[now]=build(l,mid);
    rs[now]=build(mid+1,r);
    return now;
} //初始时的线段树
in int add(int k,int l,int r,int x,int t)
{
    int now=++tot;
    if(l==r)
    {
        val[now]=t;
        ls[now]=rs[now]=0;
        return now;
    } //到了目标点,修改它
    ls[now]=ls[k],rs[now]=rs[k];
    int mid=(l+r)>>1;
    if(x<=mid) ls[now]=add(ls[now],l,mid,x,t); //若目标点在原树的左子树上,则新建左儿子
    else rs[now]=add(rs[now],mid+1,r,x,t); //若在右儿子上,同理
    return now;
} //修改并添加新路径
in int query(int k,int l,int r,int x)
{
    if(l==r) return val[k];
    int mid=(l+r)>>1;
    if(x<=mid) return query(ls[k],l,mid,x);
    else return query(rs[k],mid+1,r,x);
} //查询
int main()
{
    n=read(),m=read();
    for(re int i=1;i<=n;i++)
        a[i]=read();
    root[0]=build(1,n);
    for(re int i=1;i<=m;i++)
    {
        int v=read(),o=read();
        if(o==1)
        {
            int x=read(),y=read();
            root[i]=add(root[v],1,n,x,y);
        }
        else
        {
            int x=read();
            cout<<query(root[v],1,n,x)<<endl;
            root[i]=root[v];
        }
    }
    /*for(re int i=0;i<=10;i++)
    {
        cout<<"case #"<<i<<":    ";
        for(re int j=1;j<=n;j++)
        cout<<query(root[i],1,n,j)<<' ';
        cout<<endl;
    }//打印每个历史版本 */ 
    return 0;
}
/*
 9.30 By yzhx
*/

权值线段树&&可持久化线段树&&主席树

原文:https://www.cnblogs.com/yzhx/p/11615616.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!