线段树初步__ZERO__.
线段树,顾名思义,就是指一个个线段组成的树。
线段树的定义就是:
线段树是一种二叉搜索树,与区间树相似,它将一个区间划分成一些单元区间,每个单元区间对应线段树中的一个叶结点。使用线段树可以快速的查找某一个节点在若干条线段中出现的次数,时间复杂度为O(logN)。
——摘自百度百科
如图,这就是一棵线段树:
那么线段树有哪些神奇的性质呢?
1.它是一棵满二叉树
2.每一个节点(一段区间)是它的两个子节点的并或最大值(RMQ)。
3.(因为线段树是满二叉树)树的空间复杂度是O(4N);
线段树例题
线段树的树结构:
此处我用一个一维的数组seg[node]表示节点为node所划分到的区间的值。
线段树的建树:
因为线段树是一棵满二叉树,所以可以采用递归的方式来实现储存。
void build(int l,int r,int node)
{
if(l==r)seg[node]=a[l];//a[i]表示读入数列的第i个数
else
{
int mid=(l+r)>>;
build(l,mid,node*);
build(mid+,r,node*+);
up(node);//见下
}
return ;
}
Bulid
此处需要介绍一个up(node)函数。
这是用于对这个节点的区间值进行更新。
void up(int node){seg[node]=seg[node*]+seg[node*+];}
Up
建完树,但我们还需要做一些其他的操作,比如说区间查询(区间求和)或区间修改等。
所以我们就先介绍区间查询。
//L、R:目前访问区间的左右节点,ql,qr:查询区间的左右节点,node:当前节点的编号。
int query(int l,int r,int ql,int qr,int node)
{
if(l>=ql&&r<=qr)return seg[node];//①
else
{
int mid=(l+r)>>;
down(mid-l+,r-mid,node);//②
int ans=;
if(ql<=mid)ans+=query(l,mid,ql,qr,node*);//③
if(qr>mid) ans+=query(mid+,r,ql,qr,node*+);
return ans;
}
}
Query
①:因为线段树的节点表示一段区间,所以只要找到访问区间在查询区间内,就可以直接返回这段区间的值,不需要继续递归下去。
②:
{
这是个需要介绍的东西 称为Lazy标记。这是个很重要的东西,线段树的核心之一。
此处需要开一个add[node]数组,表示下标为node的节点需要加上add[node]的Lazy标记。这样就可以完成下放标记的任务。
void down(int l,int r,int node)
{
if(add[node]!=)//KC
{
add[node*]+=add[node];//向左子树下放标记
add[node*+]+=add[node];
seg[node*]+=add[node]*l;//④
seg[node*+]+=add[node]*r;
add[node]=;
}
return ;
}
Down
④:此处的l为上query函数的mid-l+1,为node节点的左子树的区间长度。之所以把 node左子树+Lazy标记*左子树区间长度 是因为每一个叶节点都需要加上此节点的Lazy标记。r处同上。
}
③:这需要解决的是为什么ql≤mid就可以直接ans+=query(node*2)。这主要是因为我们接下去寻找的是当前L~Mid区间里的在查询范围内的节点,因为ql≤mid无非就两种情况,l<ql或l≥ql且r在此情况下都大于等于ql,又因为每次return回来的一定是查询区间范围内的值,所以只要ql≤mid就可以了。qr>mid同上。
下面是区间修改;
//v为区间内修改(增加或减少)的值
void change(int l,int r,int ql,int qr,int node,int v)
{
if(l>=ql&&r<=qr)
{
seg[node]+=v*(r-l+);//①
add[node]+=v;//①
return ;
}
else
{
int mid=(l+r)>>;
down(mid-l+,r-mid,node);
if(ql<=mid)change(l,mid,ql,qr,node*,v);
if(qr>mid) change(mid+,r,ql,qr,node*+,v);
up(node);//②
}
}
Change
①:如果访问区间已经在修改区间内,就可以直接修改node节点的值,并在node节点处留下标记,等着下一次询问或修改的时候做下放标记的操作。r-l+1为访问区间的长度。
②:因为此处为修改操作,需要进行up函数来更新。
线段树初步__ZERO__.的更多相关文章
- 线段树初步——转载自ljc20020730
线段树初步 线段树模板1:https://www.luogu.org/problem/show?pid=3372 线段树模板2:https://www.luogu.org/problem/show ...
- 线段树初步&&lazy标记
线段树 一.概述: 线段树是一种二叉搜索树,与区间树相似,它将一个区间划分成一些单元区间,每个单元区间对应线段树中的一个叶结点. 对于线段树中的每一个非叶子节点[a,b],它的左儿子表示的区间为[a, ...
- 线段树分治初步学习&洛谷P5227[AHOI2013]连通图
线段树分治 其实思想说起来是比较简单的,我们把这个题里的所有操作(比如连边删边查询balabala)全部拍到一棵线段树上,然后对着整棵树dfs一下求解答案,顺便把操作做一下,回溯的时候撤销一下即可.虽 ...
- 【BZOJ-3779】重组病毒 LinkCutTree + 线段树 + DFS序
3779: 重组病毒 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 224 Solved: 95[Submit][Status][Discuss] ...
- 线段树(单点更新)HDU1166、HDU1742
在上一篇博文里面,我提到了我不会线段树,现在就努力地学习啊! 今天AC一题感觉都很累,可能是状态不佳,在做HDU1166这题目时候,RE了无数次. 原因是:我的宏定义写错了,我已经不是第一犯这种错误了 ...
- Luck and Love(二维线段树)
Luck and Love Time Limit: 10000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Tota ...
- poj_2528Mayor's posters(线段树)
poj_2528Mayor's posters(线段树) 标签: 线段树 题目连接 Mayor's posters Time Limit: 1000MS Memory Limit: 65536K To ...
- BZOJ4695 最假女选手(势能线段树)
BZOJ题目传送门 终于体会到初步掌握势能分析思想的重要性了. 一开始看题,感觉套路还是很一般啊qwq.直接在线段树上维护最大值和最小值,每次递归更新的时候,如果不能完全覆盖就暴力递归下去.挺好写的欸 ...
- 【洛谷P4097】Segment 李超线段树
题目大意:维护一个二维平面,给定若干条线段,支持询问任意整数横坐标处对应的纵坐标最靠上的线段的 id,相同高度取 id 值较小的,强制在线. 题解:初步学习了李超线段树.李超线段树的核心思想在于通过标 ...
随机推荐
- [js高手之路]HTML标签解释成DOM节点
最近在封装一个开源框架,已经写了500行, 已经具备jquery的大多数常用功能.跟jquery的使用方法完全一样,jquery的选择器,几乎都能支持,为什么说这事,跟这篇文章的主题有毛关系呢?因为这 ...
- Sublime Text3自定义代码片段
1.打开工具--插件开发--新建代码片段 会出现下图: 2.在<![CDATA[和]]>内写下你要的代码片段,注意的是代码片段要靠最左边. 3.设置快捷键,把下面tabTrigger标签的 ...
- [2013-01-15]The Little Schemer 学习笔记
<The Little Schemer> FP编程.lisp入门必备 这书貌似没中文版: 有英文pdf版:完整版下载链接 英文不好的,被前几页噎住的,可以先到这里看翻译好的前言部分 看完人 ...
- POI处理Excel中的日期数据类型
在POI处理Excel中的日期类型的单元格时,如果仅仅是判断它是否为日期类型的话,最终会以NUMERIC类型来处理. 正确的处理方法是先判断单元格 的类型是否则NUMERIC类型, 然后再判断单元格是 ...
- 安装mariadb二进制程序
author:JevonWei 版权声明:原创作品 下载mariadb软件包 https://downloads.mariadb.org/mariadb/5.5.57/ 一.创建用户和准备数据目录 1 ...
- Ajax异步请求模板
$.ajax({ url: '', type: 'post', data: {'id':id}, dataType: 'json', success: function(data,statusText ...
- 交换基本数据类型的方法swap,并影响到主方法
不知道朋友在哪里看到的问题,qq来问我,题目是:在不修改主方法的前提下使用一个方法交换两个int的值,方法如下: public static void main(String[] args) { In ...
- 集美大学网络1413第七次作业成绩(团队三) --需求改进&系统设计
题目 团队作业3--需求改进&系统设计 团队作业3成绩 团队/分值 TD BZ GJ CJ SI WBS GS JG DB SS SJ CS DC 总分 1 0.25 0.75 1 0.5 ...
- 个人作业3——个人总结(Alphe)
小结: 1.软件工程的第一阶段终于结束了,说实话,每个人的课程都很紧张,在这么紧张的时期我们都每周抽出一些时间来开个小会总结或者计划软件工程的相关任何非常难得,大家的态度都诚恳认真,我亦是如此,只是我 ...
- 201521123012 《Java程序设计》第八周学习总结
1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结集合与泛型相关内容. 1.2 选做:收集你认为有用的代码片段 2. 书面作业 1.本次作业题集集合 List中指定元素的删除(题目4 ...