关于离散化权值线段树和动态开点的记录

Posted 吃花椒的妙酱

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了关于离散化权值线段树和动态开点的记录相关的知识,希望对你有一定的参考价值。

ps:以下说的"区间"除非强调是权值区间,否则都是下标区间的意思

本文主要记录下,主xi树中一些曾让我困惑的地方

一个疑惑是插入的写法,额还有关于是否要copy旧结点的问题

关于插入数据
1,插入时候,两种更新,一种是pushup形式(就是普通线段树形式),一种是从上而下直接更新(lazy标记永久化写法里好像有这样写过)

//插入写法一,递归到叶子,然后pushup更新sum
void update(int pre,int &now,int l,int r ,int val)

    now = ++tot;//每个点必定是新开的
    lch[now] = lch[pre];
    rch[now] = rch[pre];
    // cout<<" now lch rch "<<now<<" "<<l<<" "<<r<<endl;
    if(l==r ) 
    
        sum[now] = sum[pre] + 1;
        return;
    
    int mid = (l+r)>>1;
    if( val <= mid ) update(lch[pre],lch[now],l,mid,val);
    else update(rch[pre],rch[now],mid+1,r,val);
    sum[now] = sum[lch[now]] + sum[rch[now]];//pushup
//插入的写法二,自上而下地更新sum
void update(int pre,int &now,int l,int r ,int val)

    now = ++tot;//每个点必定是新开的
    lch[now] = lch[pre];
    rch[now] = rch[pre];
    sum[now] = sum[pre] + 1;
    if(l==r ) 
    
        // sum[now] = sum[pre] + 1;
        return;
    
    int mid = (l+r)>>1;
    if( val <= mid ) update(lch[pre],lch[now],l,mid,val);
    else update(rch[pre],rch[now],mid+1,r,val);
    // sum[now] = sum[lch[now]] + sum[rch[now]];//pushup


2,在学树套树的时候,发现树状数组维护的线段树,无需copy。

我的理解是,两个原因,对于一次查询,一个是因为树状数组维护的是log个区间不再是[1,i]而是lowbit~i,还有一个是修改是在线的。
对于区间lowbit~i维护的[1,1e9],我们可以直接修改权值区间,无需copy上一个版本
而求区间第k大那题,为什么需要copy旧版本呢,因为维护的是一个前缀线段树——为了保证[1,i]的信息完整,我们需要将现在改好的一个log长的链,拼到上一个树(即[1,i-1])上以知道[1,i]的信息。
那为什么树套树可以不用呢,因为我们的在线修改就是改的树状数组维护的整个区间lowbit~i,相当于区间第k大那题,我们每次只问1~n区间里第k大的树,那么我们只要维护[1,n]的信息,无需知道任意[1,i]的信息了,两者是等价的。如果要copy的话,也就是我们有知道lowbit~i-1,lowbit~i-2等区间的需求,显然没有...

还有一个疑惑是关于动态开点的定义

问了下庄哥,好像上面所提及的不离散,直接维护[0,1e9]的信息,不算动态开点?就相当于把内存都先开满了N*32,离散化就是可以卡卡常吧
庄哥说他理解的是,用的时候再去申请,一般是带修的问题?更透彻的理解等我再多做点,再补充吧,太菜了现在。
不过不离散的话内存消耗真的很大,带修主席树那题卡空间卡掉了我的define int long long 

n,q=1e5,[0,1e9]离散化后[1,2e5]空间常数从N*30*5到N*18*5还是有点大的

写到这里,临时想起一个疑问就是关于root所维护的区间

主席树模板题,root[i]维护的是[1,i]的根节点,可持久化数组模板题,root[i]维护的是每次修改时的根节点,对于原始数组形成的整棵线段树是存在root[0]里的。。。似乎是看有需求查询的历史版本的时间戳??

有大佬知道话,麻烦解答一下qaq

以上是关于关于离散化权值线段树和动态开点的记录的主要内容,如果未能解决你的问题,请参考以下文章

线段树 动态开点

[CSP-S模拟测试]:表格(动态开点二维线段树+离散化)

权值线段树

HFUUOJ1024 动态开点线段树+标记永久化

#RMQ,动态开点线段树#CF803G Periodic RMQ Problem

Luogu P1637 三元上升子序列