也谈线段树

国内大佬们写的很难理解,找了个外国友人的文章,一下就看懂了。本文参考:
geeksforgeeks基础线段树
geeksforgeeks懒标记区间更新
要掌握线段树,得一步一步来。一上来就lazytag,很难理解。

一、普通单点修改

如果修改的单点属于当前树上节点覆盖的范围,直接改,然后改左右子树。没有什么pushup和pushdown。

//ss、se分别是当前树上节点覆盖范围开始和结束下标
//si是树上元素在树的数组里的下标,i是原数组下标,diff是加多少
//调用的时候从根开始update(1,n,1,5,20)
void update(int ss, int se,, int si, int i,  int diff) 
{ 
    if (i < ss || i > se) 
        return; 
    st[si] = st[si] + diff; 
    if (se != ss) 
    { 
        int mid = getMid(ss, se); 
        update(ss, mid,  2*si, i, diff); 
        update(mid+1, se,,2*si+1, i, diff); 
    } 
}

二、普通区间修改

区间修改,先看树上节点覆盖的范围和修改的范围有没有交集,没有就什么都不干;有的话分两种情况,一是到了叶子,直接更新;二是没到叶子节点,又分两种情况,1是节点覆盖范围被修改范围完全覆盖;2是不完全覆盖,不管哪种情况,做法都一样,直接更新左右子树,更新完以后,重新计算左右子树的值,更新当前节点值。也没有什么pushup和pushdown。

//us和ue分别是更新区间的下标开始、结束
void update(int ss, int se, int si, int us, int ue, int diff){
    if (ss > ue || se < us) return;
    if(ss == se){
        st[si].v = st[si].v + diff;
        return;
    }
    int mid = getmid(ss, se);
    update(ss, mid, si * 2, us, ue, diff);
    update(mid+1, se, si * 2 + 1, us, ue, diff);
    st[si].v = st[si*2].v + st[si*2+1].v;
}

仔细体会这种方式,类似深度优先遍历,从根直接到叶子节点,叶子节点更新完成后,一层一层往上更新中间节点,最后更新根。

三、懒标记区间修改

暴力区间修改太慢了,最坏情况下,如果更新整个数组,复杂度O(nlogn),比直接在原数组上更新还慢,所以必须改进。
改进办法是加入懒标记,首先必须明确最重要的一点,当一个树上节点覆盖范围完全被更新区间包含时,这个节点和所有这个节点的子孙都需要更新;反之如果一个树上节点覆盖范围和更区间部分重合,则肯定有一部分子孙需要更新,另一部分绝不需要更新。我们的做法是,该更新还是更新,直接更新就行((se-ss+1)x diff),而不像上面暴力更新那样,深度优先到叶子上,从叶子一层一层往上更新。直接更新完以后,给子孙设置懒标记,被设置懒标记的节点,先不要动,等以后更新或者查询的时候,再处理。
一个节点的懒标记,延迟的是这个节点和它的所有子孙的更新。当一个节点遇到更新和查询操作时,有懒标记的话就先消化懒标记,然后把懒标记下传(也就是他们说的pushdown)给子孙,最后正常更新。
更新完一个节点后,也需要下传懒标记,停止更新进程,把子孙的更新推迟。
两种情况需要下传懒标记,一是自己消化懒标记时,二是自己更新时。下传懒标记的时候注意判断自己是不是叶子,不是才下传,是的话下传就数组越界了。
总之,懒标记是爸爸给他的,不是自己给自己的。懒标记的消化,在更新和查询操作中。懒标记消化分3步:更新自己、传给儿子、还原初始状态(还原或清零)。
举个例子,首先更新1-3,有个节点覆盖1-3,先把它更新,懒标记下传给1-2的爸爸,和3,结束。这时要查询2-4,需要查询2和3,这两个节点上都有懒标记,先消化,再返回。
看代码:

//洛谷p3373线段树模板2
#include <cstdio>
#define MAXN 100000
typedef long long ll;
using namespace std;

//线段树节点,v表示值,lza加法懒标记,lzm乘法懒标记
struct node {
    ll v, lza, lzm; 
} st[MAXN*4+1];
int a[MAXN+1];
int n, m, p;
inline int getmid(int s, int e){
    return s + (e - s) / 2;
}
inline int left(int si){
    return si * 2;
}
inline int right(int si){
    return si * 2 + 1;
}
ll build(int ss, int se, int si){
    st[si].lzm = 1;
    if (ss == se) {
        return st[si].v = a[ss] % p;
    }
    int mid = getmid(ss, se);
    return st[si].v = (build(ss, mid, si * 2) + build(mid+1, se, si * 2 + 1)) % p;
}
void update(int ss, int se, int si, int us, int ue, int op, int opt){
    if (st[si].lzm != 1){
        st[si].v = st[si].v * st[si].lzm % p;//消化
        if(ss != se){//下传
            st[left(si)].lzm = st[left(si)].lzm * st[si].lzm % p;
            st[left(si)].lza = st[left(si)].lza * st[si].lzm % p;
            st[right(si)].lzm = st[right(si)].lzm * st[si].lzm % p;
            st[right(si)].lza = st[right(si)].lza * st[si].lzm % p;
        }
        st[si].lzm = 1;//还原
    }
    if (st[si].lza != 0){
        st[si].v = (st[si].v + (se - ss + 1) * st[si].lza) % p;
        if (ss != se){
            st[left(si)].lza = (st[left(si)].lza + st[si].lza) % p;
            st[right(si)].lza = (st[right(si)].lza + st[si].lza) % p;
        }
        st[si].lza = 0;
    }
    if (ss > ue || se < us) return;
    if (ss >= us && se <= ue){//完全在更新范围内
        //先更新自己
        if (op == 1){
            st[si].v = st[si].v * opt % p;
        } else if (op == 2){
            st[si].v = (st[si].v + (se - ss + 1) * opt) % p;
        }
        if (ss != se){//给儿孙设置懒标记
            if(op == 1){
                st[left(si)].lzm = st[left(si)].lzm * opt % p;
                st[left(si)].lza = st[left(si)].lza * opt % p;
                st[right(si)].lzm = st[right(si)].lzm * opt % p;
                st[right(si)].lza = st[right(si)].lza * opt % p;
            } else {
                st[left(si)].lza += opt;
                st[right(si)].lza += opt;
            }
        }
        return;
    }

    int mid = getmid(ss, se);
    update(ss, mid, left(si), us, ue, op, opt);
    update(mid+1, se, right(si), us, ue, op, opt);
    st[si].v = (st[left(si)].v + st[right(si)].v) % p;
}
ll query(int ss, int se, int si, int qs, int qe){
    if (st[si].lzm != 1){
        st[si].v = st[si].v * st[si].lzm % p;//消化
        if(ss != se){//下传
            st[left(si)].lzm = st[left(si)].lzm * st[si].lzm % p;
            st[left(si)].lza = st[left(si)].lza * st[si].lzm % p;
            st[right(si)].lzm = st[right(si)].lzm * st[si].lzm % p;
            st[right(si)].lza = st[right(si)].lza * st[si].lzm % p;
        }
        st[si].lzm = 1;//还原
    }
    if (st[si].lza != 0){
        st[si].v = (st[si].v + (se - ss + 1) * st[si].lza) % p;
        if (ss != se){
            st[left(si)].lza = (st[left(si)].lza + st[si].lza) % p;
            st[right(si)].lza = (st[right(si)].lza + st[si].lza) % p;
        }
        st[si].lza = 0;
    }
    if(ss >= qs && se <= qe){
        return st[si].v;
    }
    if (ss > qe || se < qs) return 0;
    int mid = getmid(ss, se);
    return (query(ss, mid, si * 2, qs, qe) + query(mid + 1, se, si * 2 + 1, qs, qe)) % p;
}
int main(){
    // freopen("P3373_2.in", "r", stdin);
    scanf("%d%d%d", &n, &m, &p);
    for(int i = 1; i <= n; i++){
        scanf("%d", a + i);
    }
    build(1, n, 1);
    int op, x, y, k;
    while(m--){
        scanf("%d", &op);
        if (op == 1 || op == 2){
            scanf("%d%d%d", &x, &y, &k);
            update(1, n, 1, x, y, op, k);
        } else {
            scanf("%d%d", &x, &y);
            printf("%lld\n", query(1, n, 1, x, y));
        }
    }
    return 0;
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容

  • 一、综述 线段树之所以称为“树”,是因为其具有树的结构特性,这种特性在处理区间问题上具有极高的效率。线段树的逻辑结...
    pidastar阅读 2,562评论 0 0
  • 这应该是系统介绍LC的线段树题目全网截止发文时最全的文章了。从这篇文章里,你可以学到如何用线段树思维和模板解LC的...
    西部小笼包阅读 1,432评论 0 1
  • 第三篇线段树了——重点不在于解决题目,通过题目理解线段树才是重点 前面写了一篇关于线段树的单点更新,线段树的单点更...
    徐森威阅读 3,060评论 0 0
  • 转自:http://www.cnblogs.com/TenosDoIt/p/3453089.html#b 一 概述...
    碧影江白阅读 1,542评论 1 2
  • https://www.cnblogs.com/jason2003/p/9676729.html已知一个数组,对其...
    素理想阅读 113评论 0 0