线段树(Segment Tree)

入门模板题 洛谷oj P3372

题目描述

  如题,已知一个数列,你需要进行下面两种操作:

  1.将某区间每一个数加上x

  2.求出某区间每一个数的和

输入格式

  第一行包含两个整数N、M,分别表示该数列数字的个数和操作的总个数。

  第二行包含N个用空格分隔的整数,其中第i个数字表示数列第i项的初始值。

  接下来M行每行包含3或4个整数,表示一个操作,具体如下:

  操作1: 格式:1 x y k 含义:将区间[x,y]内每个数加上k

  操作2: 格式:2 x y 含义:输出区间[x,y]内每个数的和

输出格式

  输出包含若干行整数,即为所有操作2的结果。

输入样例

5 5

1 5 4 2 3

2 2 4

1 2 3 2

2 3 4

1 1 5 1

2 1 4

输出样例

11

8

20

数据范围

对于30%的数据:N<=8,M<=10

对于70%的数据:N<=1000,M<=10000

对于100%的数据:N<=100000,M<=100000

  才学会写线段树不久……随便扯一篇笔记orz

0x00 线段树概念

  线段树是一种二叉搜索树。它将一个区间划分成一些单元区间,每个单元区间对应线段树中的一个叶结点。

  插入(删除)操作的时间复杂度为O(logn)。

0x01 树形结构及建树

  作为一棵树,线段树的结构体大概是这样的:

struct SegmentTree{        //树的结构体
long long l,r; //覆盖的区间的左右指针
long long value; //该节点上维护的值
long long tag; //lazy标记,下面会写到,这也是线段树的精髓
}node[MAX_N*+];

  要想建一棵线段树,可以利用线段树是个二叉树的性质,进行递归建树:

void build(REG long long p,REG long long l,REG long long r){    //递归建树
node[p].l=l,node[p].r=r;
if (l==r){ //访问到了最底部,不可再分
node[p].value=read();
return ;
}
long long mid=(l+r)>>;
build(p<<,l,mid);
build((p<<)+,mid+,r);
node[p].value=node[p<<].value+node[(p<<)+].value;
}

  

  这样我们就拥有了一棵树。

0x02 lazy标记(懒标记)

  上面提到过,线段树的精髓是lazy标记。不管是要查询还是更改数列,都绕不开它。

  如果要修改一段区间(比如同时都增加k),最容易想到的方法是暴力枚举,一个个修改。这样操作m次,最坏时间复杂度是O(mn)。显然是过不去的。

  所以我们可以想:如果不修改那么多节点呢?很容易想到,我们只关心查询需要用到的节点——并不是每一个被修改的节点都会在查询中被访问到的。打个比方:假设只有一次修改,一次询问。修改修改[1,16],而询问只关心[1,8],那么我们对于[9,16]的修改都是没有意义的。

  可以想到,如果点i的两个儿子都要同时增加k,考虑先不修改两个儿子,而给点i打一个标记,标记一下i的两个儿子都要被修改。

  如果在下面的询问里要访问到i的两个儿子之一,我们再对i的两个儿子进行修改(或下传标记),再把标记归零。如果两个儿子都会被访问到,就直接取i的值(i点维护的value值是两个儿子节点的和)。

  这差不多就是lazy标记的思想,时间复杂度就降到了O(mlogn)。

  写一个下传标记的函数:

void push_down(REG const long long& p){
node[p<<].value+=(node[p].tag*(node[p<<].r-node[p<<].l+));
node[(p<<)+].value+=(node[p].tag*(node[(p<<)+].r-node[(p<<)+].l+));
node[p<<].tag+=node[p].tag;
node[(p<<)+].tag+=node[p].tag;
node[p].tag=;
}

  按照这样的思想,我们就可以写出线段树了。其他的在最终代码里写了,不再赘述。

0x03 AC代码

#include <cstdio>
#define REG register
#define MAX_N 100000 using namespace std; long long n,m;
long long a[MAX_N+]; long long ch,x,y,z; inline long long read(){ //快速读入
REG long long ch=getchar(),x=,f=;
while (ch<''||ch>''){
if (ch=='-') f=-;
ch=getchar();
} while (ch<=''&&ch>=''){
x=x*+ch-'';
ch=getchar();
}return x*f;
} struct SegmentTree{ //树的结构体
long long l,r;
long long value;
long long tag;
}node[MAX_N*+]; void build(REG long long p,REG long long l,REG long long r){ //递归建树
node[p].l=l,node[p].r=r;
if (l==r){ //访问到了最底部,不可再分
node[p].value=read();
return ;
}
long long mid=(l+r)>>;
build(p<<,l,mid);
build((p<<)+,mid+,r);
node[p].value=node[p<<].value+node[(p<<)+].value;
} void push_down(REG const long long& p){
node[p<<].value+=(node[p].tag*(node[p<<].r-node[p<<].l+));
node[(p<<)+].value+=(node[p].tag*(node[(p<<)+].r-node[(p<<)+].l+));
node[p<<].tag+=node[p].tag;
node[(p<<)+].tag+=node[p].tag;
node[p].tag=; //标记归零
} void change(REG long long p,REG const long long& x,REG const long long& y,REG long long& z){
if (x<=node[p].l&&y>=node[p].r){ //是否完全在区间内
node[p].value+=(z*(node[p].r-node[p].l+));
node[p].tag+=z;
return ;
}
if (node[p].tag) push_down(p); //不完全在区间内,如果有lazy标记,下传
REG long long mid=(node[p].l+node[p].r)>>;
if (x<=mid) change(p<<,x,y,z);
if (y>mid) change((p<<)+,x,y,z);
node[p].value=node[p<<].value+node[(p<<)+].value;
} inline long long ask(REG long long p,REG const long long& x,REG const long long& y){
if (x<=node[p].l&&y>=node[p].r) return node[p].value;
push_down(p);
REG long long mid=(node[p].l+node[p].r)>>;
long long ans=;
if (x<=mid) ans+=ask(p<<,x,y);
if (y>mid) ans+=ask((p<<)+,x,y);
return ans;
} int main(){
// freopen("test1.txt","r",stdin);
n=read(),m=read();
build(,,n);
REG long long ch;
while (m--){
ch=read();
if (ch==){
x=read(),y=read(),z=read();
change(,x,y,z);
}
else{
x=read(),y=read();
printf("%lld\n",ask(,x,y));
}
}
return ;
}

  如果写的有哪里不对,欢迎指出,轻喷orz

[参考百度百科及网友讲解]

线段树基本操作(Segment Tree)的更多相关文章

  1. 线段树(Segment Tree)总结

    0 写在前面 怎么说呢,其实从入坑线段树一来,经历过两个阶段,第一个阶段是初学阶段,那个时候看网上的一些教学博文和模板入门了线段树, 然后挑选了一个线段树模板作为自己的模板,经过了一点自己的修改,然后 ...

  2. 【数据结构】线段树(Segment Tree)

    假设我们现在拿到了一个非常大的数组,对于这个数组里面的数字要反复不断地做两个操作. 1.(query)随机在这个数组中选一个区间,求出这个区间所有数的和. 2.(update)不断地随机修改这个数组中 ...

  3. 第二十九篇 玩转数据结构——线段树(Segment Tree)

          1.. 线段树引入 线段树也称为区间树 为什么要使用线段树:对于某些问题,我们只关心区间(线段) 经典的线段树问题:区间染色,有一面长度为n的墙,每次选择一段墙进行染色(染色允许覆盖),问 ...

  4. 线段树:Segment Tree(单点修改/区间修改模板) C++

    线段树是非常有效的数据结构,可以快速的维护单点修改,区域修改,查询最大值,最小值等功能. 同时,它也很重要.如果有一天比赛,你卡在了一道线段树模板题目上,这就真的尴尬了.不过,随着时代的进步,题目也越 ...

  5. 线段树(segment tree)

    线段树是一种二叉搜索树,它的每一个结点对应着一个区间[L, R],叶子结点对应的区间就是一个单位区间,即L == R.对于一个非叶子结点[L, R],它的左儿子所表示的区间是[L, (L +R)/2] ...

  6. 【转】Senior Data Structure · 浅谈线段树(Segment Tree)

    本文章转自洛谷 原作者: _皎月半洒花 一.简介线段树 ps: _此处以询问区间和为例.实际上线段树可以处理很多符合结合律的操作.(比如说加法,a[1]+a[2]+a[3]+a[4]=(a[1]+a[ ...

  7. 线段树(segment tree )

    http://www.cnblogs.com/TenosDoIt/p/3453089.html 写的非常好! 很多学校流行一种比较的习惯.老师们很喜欢询问,从某某到某某当中,分数最高的是多少. 这让很 ...

  8. 数据结构--树状数组&&线段树--基本操作

    随笔目的:方便以后对树状数组(BIT)以及基本线段树的回顾 例题链接:http://acm.hdu.edu.cn/showproblem.php?pid=1166 例题:hdu 1166 敌兵布阵 T ...

  9. 李超线段树(segment[HEOI2013]-洛谷T4097)

    (neng了好久好久才糊弄懂得知识点...) 一.李超线段树 在线动态维护一个二维平面直角坐标系, 支持插入一条线段, 询问与直线x = x0相交的所有线段中,交点y的最大/小值 (若有多条线段符合条 ...

随机推荐

  1. 前端传递对象列表到WebApi

    public Int64 objectPOC(JObject jsonWrapper) { dynamic jsonValues = jsonWrapper; JArray jsonInput = j ...

  2. xib下这种方式创建cell

      这种方法在iOS5.0之前是不能够创建成功的.   MEConvertListTableViewCell *cell = [tableView dequeueReusableCellWithIde ...

  3. Codeforces758D Ability To Convert 2017-01-20 10:29 231人阅读 评论(0) 收藏

    D. Ability To Convert time limit per test 1 second memory limit per test 256 megabytes input standar ...

  4. polymer-quick tour of polymer

    注册一个元素 <link rel="import" href="bower_components/polymer/polymer.html">//没 ...

  5. acm.njupt 1001-1026 简单题

    点击可展开上面目录 Acm.njupt 1001-1026简单题 第一页许多是简单题,每题拿出来说说,没有必要,也说不了什么. 直接贴上AC的代码.初学者一题题做,看看别人的AC代码,寻找自己的问题. ...

  6. npm是干什么的?

    允许用户从NPM服务器下载别人编写的第三方包到本地使用. 允许用户从NPM服务器下载并安装别人编写的命令行程序到本地使用. 允许用户将自己编写的包或命令行程序上传到NPM服务器供别人使用. 其实npm ...

  7. 18-10-30 Scrum Meeting 2

    目录 站立式会议 工作记录 昨天完成的工作 1 主要完成了单词简单释义浏览和单词详细释义浏览的功能 并且已经测试和上传eolinker 2 3 主要搭建起爬虫的框架平台,并且测试了py连接服务器的功能 ...

  8. 简单配置vps,防ddos攻击

    防人之心不可无. 网上总有些无聊或者有意的人.不多说了.上干货,配置vps apf防小流量ddos攻击. 对于大流量的ddos攻击, 需要机房的硬件防火墙,vps内部可能也扛不住. 1. 安装 DDo ...

  9. MVP社区巡讲 12月5日北京站| 12月12日上海站

    2015年底的社区巡讲Powered MVP Roadshow正式启动啦!12月5日周六下午北京场,12月12日周六下午上海场. 欢迎各位邀请您的同事朋友来参加MVP的社区活动,也邀请您发送活动信息( ...

  10. TFS 2015新功能之一,当前迭代查询标记

    TFS 2015发布在即,有幸作为MVP提前获得了TFS的RTM版本,下面就TFS 2015的新功能做一些介绍:   TFS 2015新功能之一,当前迭代查询标记 在TFS的查询中,可以将" ...