SGTtrick

By 蒟蒻 ldxoiBy\ 蒟蒻\ ldxoiBy 蒟蒻 ldxoi

Chapter 1.关于线段树操作的一些分析

我们知道,线段树有两个核心的函数pushdownpushdownpushdown和pushuppushuppushup。

以及两类对于一段区间进行操作的函数updateupdateupdate和queryqueryquery

博主在这里简单讲一下几个函数的功能:

首先我们假设用ValValVal表示维护信息的类型,TagTagTag表示懒标记的类型。

显然一个线段树节点(类型为NodeNodeNode)是由一个ValValVal元素和一个TagTagTag以及表示区间管辖范围的l,rl,rl,r构成的。

struct Tag{...};
struct Val{...};
struct Node{Val val;Tag tag;int l,r;}T[N<<2];
  • pushdown(int p)pushdown(int\ p)pushdown(int p)表示将ppp的标记下传给ppp的儿子并更新它们的信息和标记
  • pushup(int p)pushup(int\ p)pushup(int p)表示把ppp儿子的信息合并成ppp的信息
  • query(int p,int ql,int qr)query(int\ p,int\ ql,int\ qr)query(int p,int ql,int qr)表示查询区间[ql,qr][ql,qr][ql,qr]的一些信息
  • update(int p,int ql,int qr,Tag v)update(int\ p,int\ ql,int\ qr,Tag\ v)update(int p,int ql,int qr,Tag v)表示用标记vvv对区间[ql,qr][ql,qr][ql,qr]的信息和标记进行更新

可以看出来这些函数分成两类:

  • 父亲朝儿子转移:update,pushdownupdate,pushdownupdate,pushdown
  • 儿子朝父亲转移:query,pushupquery,pushupquery,pushup

而我们继续将这几个函数的功能进行拆分并取上并集(雾,发现它其实是这几个东西的组合:

  • 两个ValValVal类型的元素的合并
  • 一个TagTagTag类型的元素对一个ValValVal类型的元素的更新
  • 一个TagTagTag类型的元素对一个TagTagTag类型的元素的更新
inline Tag operator+(const Tag&a,const Tag&b){...}
inline void operator+=(Tag&a,const Tag&b){a=a+b;}
inline Val operator+(const Val&a,const Tag&b){...}
inline void operator+=(Val&a,const Tag&b){a=a+b}
inline Val operator+(const Val&a,const Val&b){...}
inline void operator+=(Val&a,const Val&b){a=a+b;}

如果我们对这几个操作重载运算符的话,那么上述函数的实现就很简单了,为了让实现更加简便可以设计一个pushnow函数pushnow函数pushnow函数。

  • pushuppushuppushup函数:把儿子的ValValVal合并成自己的ValValVal。
inline void pushup(int p){
    T[p].val=T[lc].val+T[rc].val;
}
  • pushnowpushnowpushnow函数:pushnow(int p,Tag v)pushnow(int\ p,Tag\ v)pushnow(int p,Tag v)表示用标记vvv更新节点ppp的信息和标记。

    代码:
inline void pushnow(int p,Tag v){
    T[p].val+=v,T[p].tag+=v;
}
  • pushdownpushdownpushdown函数:用自己的TagTagTag去更新儿子的ValValVal和TagTagTag
inline void pushdown(int p){
    if(check(T[p].tag))return;
    pushnow(lc,T[p].tag),pushnow(rc,T[p].tag);
    T[p].tag=tag_empty;
}
  • queryqueryquery函数:把要查询的区间拆成线段树上至多logloglog个区间把它们的ValValVal按一定顺序合并起来。
inline Val query(int p,int ql,int qr){
    if(ql>T[p].r||qr<T[p].l)return val_empty;
    if(ql<=T[p].l&&T[p].r<=qr)return T[p].val;
    pushdown(p);
    if(qr<=mid)return query(lc,ql,qr);
    if(qr>mid)return query(rc,ql,qr);
    return query(lc,ql,qr)+query(rc,ql,qr);
}
  • updateupdateupdate函数:把要修改的区间拆成线段树上至多logloglog个区间分别用给出的TagTagTag更新它们的TagTagTag和ValValVal。
inline void update(int p,int ql,int qr,Tag v){
    if(ql>T[p].r||qr<T[p].l)return;
    if(ql<=T[p].l&&T[p].r<=qr)return pushnow(p,v);
    pushdown(p);
    if(qr<=mid)update(lc,ql,qr,v);
    else if(ql>mid)update(rc,ql,qr,v);
    else update(lc,ql,qr,v),update(rc,ql,qr,v);
    pushup(p);
}

所以当你拿到一道线段树题的时候,思考上述三种运算符如何重载即可qwqqwqqwq。

那么最后我们给上一波上述所有内容合起来的伪代码:才不会告诉你这是一个通用的框架呢

namespace SGT{
    #define lc (p<<1)
    #define rc (p<<1|1)
    #define mid (T[p].l+T[p].r>>1)
    struct Tag{...};
    struct Val{...};
    const Tag tag_empty=(Tag){...};
    const Val val_empty=(Val){...};
    inline Tag operator+(const Tag&a,const Tag&b){...}
    inline void operator+=(Tag&a,const Tag&b){a=a+b;}
    inline Val operator+(const Val&a,const Tag&b){...}
    inline void operator+=(Val&a,const Tag&b){a=a+b}
    inline Val operator+(const Val&a,const Val&b){...}
    inline void operator+=(Val&a,const Val&b){a=a+b;}
    struct Node{Val val;Tag tag;int l,r;}T[N<<2];
    inline bool check(const Tag&v){...}
    inline void pushdown(int p){
        if(check(T[p].tag))return;
        T[lc].val+=T[p].tag,T[lc].tag+=T[p].tag;
        T[rc].val+=T[p].tag,T[rc].tag+=T[p].tag;
        T[p].tag=tag_empty;
    }
    inline void pushup(int p){
        T[p].val=T[lc].val+T[rc].val;
    }
    inline void build(int p,int l,int r){
        T[p]=(Node){val_empty,tag_empty,l,r};
        if(l==r){
            T[p].val=(Val){...};
            return;
        }
        build(lc,l,mid);
        build(rc,mid+1,r);
        pushup(p);
    }
    inline void update(int p,int ql,int qr,Tag v){
        if(ql>T[p].r||qr<T[p].l)return;
        if(ql<=T[p].l&&T[p].r<=qr){
            T[p].val+=v,T[p].tag+=v;
            return;
        }
        pushdown(p);
        if(qr<=mid)update(lc,ql,qr,v);
        else if(ql>mid)update(rc,ql,qr,v);
        else update(lc,ql,qr,v),update(rc,ql,qr,v);
        pushup(p);
    }
    inline Val query(int p,int ql,int qr){
        if(ql>T[p].r||qr<T[p].l)return val_empty;
        if(ql<=T[p].l&&T[p].r<=qr)return T[p].val;
        pushdown(p);
        if(qr<=mid)return query(lc,ql,qr);
        if(qr>mid)return query(rc,ql,qr);
        return query(lc,ql,qr)+query(rc,ql,qr);
    }
    #undef lc
    #undef rc
    #undef mid
}

Chapter 2.线段树的两种写法

吐槽:我吐我自己

这里简单谈一谈线段树的两种实现方法:dfsdfsdfs版和bfsbfsbfs版,其中后者为我瞎yyyyyy的,经过测试实际效果跟dfsdfsdfs版差距不大。

我才不会告诉你们我测出来很多题用第二种写法要慢一些呢

好吧我承认我发现的这种bfsbfsbfs版写法很鸡肋。

正题

众所周知,搜索有几种顺序,其中有两种很著名分别叫做dfsdfsdfs和bfsbfsbfs。

而我们的常规线段树就是使用的dfsdfsdfs序。

然而在博主的努力尝试下,bfsbfsbfs序也是可以处理的。

为什么呢?

因为在Chapter 1Chapter\ 1Chapter 1中我们已经谈到过线段树的queryqueryquery和updateupdateupdate操作的本质了:

  • queryqueryquery函数:把要查询的区间拆成线段树上至多logloglog个区间把它们的ValValVal按顺序合并起来。
  • updateupdateupdate函数:把要修改的区间拆成线段树上至多logloglog个区间分别用给出的TagTagTag更新它们的TagTagTag和ValValVal。

这样的话,只要按照顺序把这logloglog段区间找到再合起来一定就可以维护所有的操作。

因为按照bfsbfsbfs版本的顺序来合并也是正确的。

如何维护顺序?用一个队列即可。

于是我们可以给出bfsbfsbfs版的一些框架,这里跟dfsdfsdfs版本相同的函数就不拿上来了。

代码如下:

namespace SGT{
    ...
    int q[N<<2],hd,tl;
    inline void build(int n){
        q[hd=tl=1]=1,T[1]=(Node){val_empty,tag_empty,1,n};
        while(hd<=tl){
            int p=q[hd++];
            if(T[p].l==T[p].r){
                T[p].val=(Val){...};
                continue;
            }
            T[lc]=(Node){val_empty,tag_empty,T[p].l,mid};
            T[rc]=(Node){val_empty,tag_empty,mid+1,T[p].r};
            q[++tl]=lc,q[++tl]=rc;
        }
        for(ri i=tl;i^1;--i)T[q[i]>>1].val+=T[q[i]].val;
    }
    inline void update(int p,int ql,int qr,Tag v){
        q[hd=tl=1]=1;
        while(hd<=tl){
            int p=q[hd++];
            if(ql>T[p].r||qr<T[p].l)continue;
            if(ql<=T[p].l&&T[p].r<=qr){
                pushnow(p,v);
                continue;
            }
            pushdown(p);
            if(qr<=mid)q[++tl]=lc,T[p].val=T[rc].val;
            else if(ql>mid)q[++tl]=rc,T[p].val=T[lc].val;
            else q[++tl]=lc,q[++tl]=rc,T[p].val=val_empty;
        }
        for(ri i=tl;i^1;--i)T[q[i]>>1].val+=T[q[i]].val;
    }
    inline Val update(int p,int ql,int qr,Tag v){
        Val ret=val_empty;
        q[hd=tl=1]=1;
        while(hd<=tl){
            int p=q[hd++];
            if(ql>T[p].r||qr<T[p].l)continue;
            if(ql<=T[p].l&&T[p].r<=qr){
                ret+=T[p].val;
                continue;
            }
            pushdown(p);
            if(qr<=mid)q[++tl]=lc;
            else if(ql>mid)q[++tl]=rc;
            else q[++tl]=lc,q[++tl]=rc;
        }
        return ret;
    }
    ...
}

经过我们的努力改动,代码量成功翻了一倍!!!(滑稽

That′s allThat's\ allThat′s all

Thank youThank\ youThank you

SGTtrick的更多相关文章

随机推荐

  1. Python【每日一问】02

    问:列表 test = [1,2,3,1,3,4,5,67,7,8,54,1,2,3,4,5,6],如何删除该列表的重复元素? 方法1:利用集合的不重复性 # 利用集合的不重复性 test = [1, ...

  2. docker入门 什么是docker? 为什么使用docker?

    1.什么是docker? 轻量级操作系统虚拟化解决方案 2.为什么使用docker? 1.docker的启动是秒级的,比传统虚拟机快很多 2.资源利用率高,一台主机上可同时运行数千个docker容器 ...

  3. 01-Introspector内省机制

    在java领域编程中,内省机制相当的不错,可以省去我们程序员很多的不必要的代码 比如说:在jdbc工具类 我们可以将ResultSet结果集待到 javabean对象中 将http请求报文的数据 转换 ...

  4. 学习笔记 requests + BeautifulSoup

    第一步:requests get请求 # -*- coding:utf-8 -*- # 日期:2018/5/15 17:46 # Author:小鼠标 import requests url = &q ...

  5. 堆排序(Heapsort)

    class Program { static void Main(string[] args) { , , , , , ,}; int size = arr.Length; Heapsort(arr, ...

  6. spring 之 BeanPostProcessor

    粗略一看, 它有这么多实现: 可见, 它是多么基础 而重要的一个 接口啊! 它提供了两个方法: public interface BeanPostProcessor { Object postProc ...

  7. 03 字符串常用操作方法及For 循环

    字符串常用操作 s = 'alexWUsir' s1 = s.capitalize() #首字母大写 print(s1) #Alexwusir s2 = s.upper() #全部大写 print(s ...

  8. spark-1

    先测试搭好的spark集群: 本地模式测试: 在spark的目录下: ./bin/run-example SparkPi 10 --master local[2] 验证成功: 集群模式 Spark S ...

  9. 剑指offer例题——用两个栈实现队列

    题目:用两个栈来实现一个队列,完成队列的Push和Pop操作.队列中的元素为int类型. 首先是概念理解,栈和对列存取的区别 栈(stack)是一种后进先出(last in first out, LI ...

  10. 使用jQuery+huandlebars判断类型的helper

    兼容ie8(很实用,复制过来,仅供技术参考,更详细内容请看源地址:http://www.cnblogs.com/iyangyuan/archive/2013/12/12/3471227.html) & ...