线段树:Segment Tree(单点修改/区间修改模板) C++
线段树是非常有效的数据结构,可以快速的维护单点修改,区域修改,查询最大值,最小值等功能。
同时,它也很重要。如果有一天比赛,你卡在了一道线段树模板题目上,这就真的尴尬了。不过,随着时代的进步,题目也越来越变态,线段树更多时候则是你算法时间复杂度的优化。
这是单点查询的代码。
#include <iostream>
#include <cstdio>
#include <algorithm>
using namespace std; int n;
int a[]; struct node
{
int l;
int r;
int sum;
} t[]; void build (int i,int l,int r)
{
t[i].l=l;
t[i].r=r;
int mid=l+r>>;
if (l==r)
{
t[i].sum=a[l];return ;
} build(i<<,l,mid);
build((i<<)+,mid+,r); t[i].sum=t[i<<].sum+t[(i<<)+].sum; return ;
} void update(int i,int p,int k)
{
if (t[i].l==t[i].r)
{
t[i].sum=k;
return;
} int mid=t[i].l+t[i].r>>;
if (p<=mid) update(i<<,p,k);
if (p>mid) update((i<<)+,p,k);
t[i].sum=t[i<<].sum+t[(i<<)+].sum; return ;
} int getsum(int i,int l,int r)
{
int mid=(t[i].l+t[i].r)>>;
if (l<=t[i].l&&t[i].r<=r) return t[i].sum;
if (r<=mid) return getsum(i<<,l,r);
if (l>mid) return getsum((i<<)+,l,r); return getsum(i<<,l,r)+getsum((i<<)+,l,r);
} int main()
{
int m;
scanf("%d%d",&n,&m);
for (int i=;i<=m;i++)
{
scanf("%d",&a[i]);
}
build(,,n);
while (m--)
{
int x,y,z;
scanf("%d%d%d",&x,&y,&z);
if (x==)
{
update(,y,z);
}
else printf("%d\n",getsum(,y,z));
} return ;
}
下面是区间修改的代码。
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <vector>
#include <cmath>
#include <set>
using namespace std; struct node
{
int l,r,s,d;
} t[];
int n,m;
int a[]; void pushup(int x)
{
t[x].s=t[x<<].s+t[(x<<)+].s;
return;
} void pushdown(int x)
{
int mid=(t[x].l+t[x].r)>>;
int d=t[x].d;
t[x<<].d+=d;t[(x<<)+].d+=d;
t[x<<].s+=(mid-t[x].l+)*d;
t[(x<<)+].s+=(t[x].r-mid)*d;
return;
} void build(int i,int l,int r)
{
t[i].l=l;
t[i].r=r;
t[i].d=;
if (l==r)
{
t[i].s=a[l];
return ;
} int mid=l+r>>; build(i<<,l,mid);
build((i<<)+,mid+,r); pushup(i);
return;
} int getsum(int i,int l,int r)
{
int mid=t[i].l+t[i].r>>;
if (l<=t[i].l&&t[i].r<=r)
{
return t[i].s;
}
int d=t[i].d;
if (!d) pushdown(i);
int ret=;
if (mid>=l) ret+=getsum(i<<,l,r);
if (mid<r) ret+=getsum((i<<)+,l,r); return ret;
} void update(int i,int l,int r,int del)
{
if (t[i].l==t[i].r)
{
t[i].d+=del;
t[i].s+=(t[i].l-t[i].r+)*del;
return ;
} int mid=t[i].l+t[i].r>>;
if (!t[i].d) pushdown(i);
if (mid>=l) update(i<<,l,r,del);
if (mid<r) update((i<<)+,l,r,del);
pushup(i);
} int main()
{
scanf("%d%d",&n,&m);
for (int i=;i<=n;i++)
{
scanf("%d",&a[i]);
}
build (,,n); for (int i=;i<=m;i++)
{
int x,y,z,d;
scanf("%d",&x);
if (x==)
{
scanf("%d%d%d",&y,&z,&d);
update(,y,z,d);
}
if (x==)
{
scanf("%d%d",&y,&z);
printf("%d\n",getsum(,y,z));
}
} return ;
}
线段树:Segment Tree(单点修改/区间修改模板) C++的更多相关文章
- 『线段树 Segment Tree』
更新了基础部分 更新了\(lazytag\)标记的讲解 线段树 Segment Tree 今天来讲一下经典的线段树. 线段树是一种二叉搜索树,与区间树相似,它将一个区间划分成一些单元区间,每个单元区间 ...
- 线段树(Segment Tree)(转)
原文链接:线段树(Segment Tree) 1.概述 线段树,也叫区间树,是一个完全二叉树,它在各个节点保存一条线段(即“子数组”),因而常用于解决数列维护问题,基本能保证每个操作的复杂度为O(lg ...
- BZOJ4999:This Problem Is Too Simple!(DFS序&树上差分&线段树动态开点:区间修改单点查询)
Description 给您一颗树,每个节点有个初始值. 现在支持以下两种操作: 1. C i x(0<=x<2^31) 表示将i节点的值改为x. 2. Q i j x(0<=x&l ...
- BZOJ.4695.最假女选手(线段树 Segment tree Beats!)
题目链接 区间取\(\max,\ \min\)并维护区间和是普通线段树无法处理的. 对于操作二,维护区间最小值\(mn\).最小值个数\(t\).严格次小值\(se\). 当\(mn\geq x\)时 ...
- 【数据结构系列】线段树(Segment Tree)
一.线段树的定义 线段树,又名区间树,是一种二叉搜索树. 那么问题来了,啥是二叉搜索树呢? 对于一棵二叉树,若满足: ①它的左子树不空,则左子树上所有结点的值均小于它的根结点的值 ②若它的右子树不空, ...
- 浅谈线段树 Segment Tree
众所周知,线段树是algo中很重要的一项! 一.简介 线段树是一种二叉搜索树,与区间树相似,它将一个区间划分成一些单元区间,每个单元区间对应线段树中的一个叶结点. 使用线段树可以快速的查找某一个节点在 ...
- 线段树(segment tree)
线段树在一些acm题目中经常见到,这种数据结构主要应用在计算几何和地理信息系统中.下图就为一个线段树: (PS:可能你见过线段树的不同表示方式,但是都大同小异,根据自己的需要来建就行.) 1.线段树基 ...
- poj 2763 Housewife Wind(树链剖分+单点查询+区间修改)
题目链接:http://poj.org/problem?id=2763 题意:给一个数,边之间有权值,然后两种操作,第一种:求任意两点的权值和,第二,修改树上两点的权值. 题解:简单的树链剖分. #i ...
- 线段树 Interval Tree
一.线段树 线段树既是线段也是树,并且是一棵二叉树,每个结点是一条线段,每条线段的左右儿子线段分别是该线段的左半和右半区间,递归定义之后就是一棵线段树. 例题:给定N条线段,{[2, 5], [4, ...
- Libre OJ 130、131、132 (树状数组 单点修改、区间查询 -> 区间修改,单点查询 -> 区间修改,区间查询)
这三题均可以用树状数组.分块或线段树来做 #130. 树状数组 1 :单点修改,区间查询 题目链接:https://loj.ac/problem/130 题目描述 这是一道模板题. 给定数列 a[1] ...
随机推荐
- PHP的抽象类和接口
抽象类与接口相似,都是一种比较特殊的类.抽象类是一种特殊的类,而接口也是一种特殊的抽象类.它们通常配合面向对象的多态性一起使用.虽然声明和使用都比较容易,但它们的作用在理解上会困难一点. ①抽象类 在 ...
- 11061160_11061151_Pair Project: Elevator Scheduler软件工程结对编程作业总结
软件工程结对编程作业总结 11061160 顾泽鹏 11061151 庞梦劼 一.关于结对编程 这次的软工任务既不是单打独斗的个人任务,也不是集思广益的团队项目,而是人数为两人的结对编程.两个人合 ...
- 重定位表 IMAGE_BASE_RELOCATION
typedef struct _IMAGE_BASE_RELOCATION { DWORD VirtualAddress; DWORD SizeOfBlock; // WORD TypeOffset[ ...
- asp.net2.0app开发。
asp.net app 网站开发:http://www.cnblogs.com/TerryFeng/archive/2009/05/15/1457731.html 在OPERA或M3GATE中设置编码 ...
- Spring碎点知识
1.依赖注入:不仅可以为Bean注入普通的属性值,还可以注入其他Bean的作用.通过配置文件组织在一起,这里的Bean是Java对象 说明:关于依赖注入与控制反转的这两个名字,表达的都是同一个意思,只 ...
- 一个读取propeties配置文件的工具类,线程安全的
public class ConfigUtil { private static Map<String,Properties> map = new HashMap<String,Pr ...
- android 适配器simpleadapter和baseadapter区别
android 适配器 simpleadapter 和 baseadapter 设计网络程序或者数据处理显示程序的时候,常常会使用 simpleadapter 和baseadapter 来实现. ad ...
- Delphi:窗体自适应屏幕分辨率(根据预设值的比例改变)
delphi 程序适应屏幕分辨率,先在表单单元的Interface部分定义两个常量, 表示设计时的屏幕的宽度和高度(以像素为单位). 在表单的Create事件中先判断 当前分辨率是否与设计分辨率相同, ...
- 文件上传[Uploadify]
Uploadify文件上传插件 Uploadify简单说来,是基于Jquery的一款文件上传插件.它的功能特色总结如下: 支持单文件或多文件上传,可控制并发上传的文件数 在服务器端支持各种语言与之配合 ...
- POJ 2135 Farm Tour
题目大意:有一个无向图..农夫从1号点出发..要到达N号点..然后回到1号点..来回不能走相同的路径..问最短的距离是多少. 题解:又是不能走重复路径!基本图论算法直接扔掉上网络流.不能相同就边限1, ...