线段树之——区间修改区间查询

1.概述

线段树,也叫区间树,是一个完全二叉树,它在各个节点保存一条线段(即“子数组”),因而常用于解决数列维护问题,基本能保证每个操作的复杂度为O(lgN)。

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

对于线段树中的每一个非叶子节点[a,b],它的左儿子表示的区间为[a,(a+b)/2],右儿子表示的区间为[(a+b)/2+1,b]。因此线段树是平衡二叉树,最后的子节点数目为N,即整个线段区间的长度。

使用线段树可以快速的查找某一个节点在若干条线段中出现的次数,时间复杂度为O(logN)。而未优化的空间复杂度为2N,因此有时需要离散化让空间压缩。

2.线段树基本操作

线段树的基本操作主要包括构造线段树,区间查询和区间修改。

(1)线段树构造

首先介绍构造线段树的方法:让根节点表示区间[0,N-1],即所有N个数所组成的一个区间,然后,把区间分成两半,分别由左右子树表示。不难证明,这样的线段树的节点数只有2N-1个,是O(N)级别的,如图:

节点定义如下:

typedef struct node {
int l; //线段的左端点
int r; //线段的左端点
int value; //线段上的值
}node;

线段树的构建:

  • 伪代码

    bulid//以节点v为根建树、v对应区间为[l,r]
    {
    对节点v初始化
    if (l!=r) {
    以v的左孩子为根建树,区间为[l,(l+r)/]
    以v的右孩子为根建树,区间为[(l+r)/+,r]
    }
    }
  • 完整的建树代码

    #define N 10000
    node tree[N];
    void bulid(int l, int r, int v) //对结点v进行建立,区间为l~r
    {
    tree[v].l = l;
    tree[v].r = r;
    if(l == r) {
    //进行结点的初始化
    tree[v].value = a[r];
    return;
    }
    int mid = (l + r) / ;
    bulid(v * , l, mid);
    bulid(v * + , mid + , r);
    //根据左右儿子更新当前结点
    tree[v].value = tree[v * ].value + tree[v * + ].value;
    }

题目实现:

  • 更新

    当在a[i]~a[j]上的所有的元素都加上一个值c的时候

    如果a[i]~a[j]刚还是一个完整段的时候,直接将这个段的value值加上c*(r-l+1)

    当更新的区间不是一个完整段的时候,采用一种记录增量的方法:给每个节点增加一个域:int add,记录更新操作的增量c,初始的时候add均为0,比如当对2~5区间更新后,给该结点的add加上一个值c,再下次要对2~3结点进行更新或查询时,再将add传递到下面的孩子结点中去

    完整的更新树代码如下:

    typedef struct node {
    int l; //线段的左端点
    int r; //线段的左端点
    int value; //线段上的值
    int add;
    }node;
    void update(int v, int r, int l, int m)//更新区间l~r加上数m
    {
    if(tree[v].l == l && tree[v].r == r) { //找到,更新并记录增量
    tree[v].value += m * (r - l + );
    tree[v].add = m;
    return;
    }
    if(tree[v].add) {
    tree[ * v].add += tree[v].add;
    tree[ * v + ].add += tree[v].add;
    tree[v].add = ;
    }
    int mid = (tree[v].l + tree[v].r) / ;
    if(r <= mid) {
    update(v * , l, r, m); //只对左儿子更新
    } else {
    if(l > mid) {
    update(v * + , l, r, m); //只对右儿子更新
    } else { //区间横跨左右儿子区间,对其两者均进行更新
    update(v * , l, mid, m);
    update(v * + , mid + , r, m);
    }
    }
    }
  • 查询

    查询区间l~r上的value值

void query(int v, int l, int r)  //当前查询结点为v,要查询的区间为l~r
{
if(tree[v].l == l && tree[v].r == r) {
ans += tree[v].value;
return;
}
if(tree[v].add) {
tree[v * ].add += tree[v].add;
tree[v * + ].add += tree[v].add;
tree[v].add = ;
}
int mid = (tree[v].l + tree[v].r) / ;
if(r <= mid) {
query(v * , l, r); //要查询的区间都在左儿子
} else {
if(l > mid) {
query(v * + , l, r); //要查询的区间都在左儿子
} else { //要查询的区间横跨左右孩子
query(v * , l, mid);
query(v * + , mid + , r);
}
}
}

注:

  源地址:http://www.cnblogs.com/archimedes/p/segment-tree.html

线段树(segment_tree)的更多相关文章

  1. 并查集 + 线段树 LA 4730 Kingdom

    题目传送门 题意:训练指南P248 分析:第一个操作可以用并查集实现,保存某集合的最小高度和最大高度以及城市个数.运用线段树成端更新来统计一个区间高度的个数,此时高度需要离散化.这题两种数据结构一起使 ...

  2. 线段树(多维+双成段更新) UVA 11992 Fast Matrix Operations

    题目传送门 题意:训练指南P207 分析:因为矩阵不超过20行,所以可以建20条线段的线段树,支持两个区间更新以及区间查询. #include <bits/stdc++.h> using ...

  3. HDU 5405 (树链剖分+线段树)

    Problem Sometimes Naive 题目大意 给你一棵n个节点的树,有点权. 要求支持两种操作: 操作1:更改某个节点的权值. 操作2:给定u,v, 求 Σw[i][j]   i , j ...

  4. BZOJ 3531(树链剖分+线段树)

    Problem 旅行 (BZOJ 3531) 题目大意 给定一颗树,树上的每个点有两个权值(x,y). 要求维护4种操作: 操作1:更改某个点的权值x. 操作2:更改某个点的权值y. 操作3:求a-- ...

  5. BZOJ2243 (树链剖分+线段树)

    Problem 染色(BZOJ2243) 题目大意 给定一颗树,每个节点上有一种颜色. 要求支持两种操作: 操作1:将a->b上所有点染成一种颜色. 操作2:询问a->b上的颜色段数量. ...

  6. POJ3237 (树链剖分+线段树)

    Problem Tree (POJ3237) 题目大意 给定一颗树,有边权. 要求支持三种操作: 操作一:更改某条边的权值. 操作二:将某条路径上的边权取反. 操作三:询问某条路径上的最大权值. 解题 ...

  7. bzoj4034 (树链剖分+线段树)

    Problem T2 (bzoj4034 HAOI2015) 题目大意 给定一颗树,1为根节点,要求支持三种操作. 操作 1 :把某个节点 x 的点权增加 a . 操作 2 :把某个节点 x 为根的子 ...

  8. 线段树(区间合并) POJ 3667 Hotel

    题目传送门 /* 题意:输入 1 a:询问是不是有连续长度为a的空房间,有的话住进最左边 输入 2 a b:将[a,a+b-1]的房间清空 线段树(区间合并):lsum[]统计从左端点起最长连续空房间 ...

  9. BZOJ_1018_[SHOI2008]_交通堵塞traffic_(线段树)

    描述 http://www.lydsy.com/JudgeOnline/problem.php?id=1018 \(2*n\)的距形,起初没有边相连,之后有三种操作: 1.加边. 2.删边. 3.询问 ...

随机推荐

  1. 学号 20175201张驰 《Java程序设计》第8周学习总结

    学号 20175201张驰 <Java程序设计>第8周学习总结 教材学习内容总结 第十五章 知识总结: 1.泛型类声明:可以使用"class 名称"声明一个类,例如:c ...

  2. 爬取豆瓣电影影评,生成wordcloud词云,并利用监督学习根据评论自动打星

    本文的完整源码在git位置:https://github.com/OceanBBBBbb/douban-ml 爬取豆瓣影评 爬豆瓣的影评比较简单,豆瓣没有做限制,甚至你都不用登陆就可以看全部,我这里用 ...

  3. ajax请求网络api

    不啰嗦,直接上代码: 1.在浏览器输入网址:http://api.asilu.com/weather/?callback=getname&city=深圳 你会看到如下结果:他返回的是json数 ...

  4. Vfox数据库导出EXCEL,含有备注型子段

    1. 选择菜单“数据”-> “自其他来源”->“来自 Microsoft Query ”. 2. 在出来的“选择数据源” 里面双击第一个选项“<新数据源>”会出来一个“创建新数 ...

  5. web开发前端面试知识点目录整理

    web开发前端面试知识点目录整理 基本功考察 关于Html 1. html语义化标签的理解; 结构化的理解; 能否写出简洁的html结构; SEO优化 2. h5中新增的属性; 如自定义属性data, ...

  6. CCF CSP 201609-1 最大波动

    题目链接:http://118.190.20.162/view.page?gpid=T47 问题描述 试题编号: 201609-1 试题名称: 最大波动 时间限制: 1.0s 内存限制: 256.0M ...

  7. SlidingMenu第三篇 --- SlidingMenu使用介绍

    在Activity中通过SlidingMenu的构造方法,直接设置侧滑菜单 public class Main2Activity extends Activity { @Override protec ...

  8. TeamViewer连CentOS

    用过TeamViewer的人都会感叹其远程连接的强大,昨天有将Windows和CentOS在同一网段内互相连接打通了,今天在外网环境下突然想到是否可以用TeamViewer在外网环境下连到CentOS ...

  9. PHP环境配置遇到的小问题

    1.设置时区 2.默认打开文件 3.文件夹权限设置

  10. Java8将List转为Map

    1.实体 public class Hosting { private int id; private String name; private long websites; public Hosti ...