【线段树】【P3372】模板-线段树
Definition&Solution
线段树是一种log级别的树形结构,可以处理区间修改以及区间查询问题。期望情况下,复杂度为O(nlogn)。
核心思想见百度百科,线段树即将每个线段分成左右两个线段做左右子树。一个线段没有子树,当且仅当线段表示的区间为[a,a]。
由于编号为k的节点的子节点为2k以及2k+1,线段树可以快速的递归左右叶节点。
lazy标记:当进行区间修改的时候,如果一个区间整体全部被包含于要修改的区间,则可以将该区间的值修改后,将lazy标记打在区间上,不再递归左右区间。
例如,要修改[15,30]区间整体+2,当前区间为[16,24],被包含于要修改的区间。记代表区间[16,24]的节点编号为k,则tree[k]+=2*(24-16+1),同时lazy[k]+=2。
在下次修改或查询到k节点时,进行lazy的下放,即如下代码
inline void Free(cl l,cl r,cl p) { ll m=(l+r)>>,dp=p<<; tree[dp]+=(m-l+)*lazy[p];tree[dp+]+=(r-m)*lazy[p]; lazy[dp]+=lazy[p];lazy[dp+]+=lazy[p]; lazy[p]=; }
注意:被打上lazy标记的区间实际上已经修改完区间和,每次free修改的是子区间。
Example
Description
已知一个数列,你需要进行下面两种操作:
1.将某区间每一个数加上x
2.求出某区间每一个数的和
Input
第一行包含两个整数N、M,分别表示该数列数字的个数和操作的总个数。
第二行包含N个用空格分隔的整数,其中第i个数字表示数列第i项的初始值。
接下来M行每行包含3或4个整数,表示一个操作,具体如下:
操作1: 格式:1 x y k 含义:将区间[x,y]内每个数加上k
操作2: 格式:2 x y 含义:输出区间[x,y]内每个数的和
Output
输出包含若干行整数,即为所有操作2的结果。
Sample Input
Sample Output
Hint
时空限制:1000ms,128M
数据规模:
对于30%的数据:N<=8,M<=10
对于70%的数据:N<=1000,M<=10000
对于100%的数据:N<=100000,M<=100000
Solution
模板题。有一些需要注意的地方会在summary写明
Code
#include<cstdio> #define maxn 100010 #define maxt 400010 #define ll long long int #define cl const long long int inline void qr(long long &x) { ; ') { ; ch=getchar(); } )+(x<<)+(ch^),ch=getchar(); x*=f; return; } inline long long max(const long long &a,const long long &b) {if(a>b) return a;else return b;} inline long long min(const long long &a,const long long &b) {if(a<b) return a;else return b;} inline ) return x;else return -x;} inline void swap(long long &a,long long &b) { long long c=a;a=b;b=c;return; } ll n,m,MU[maxn],sign,a,b,c; ll tree[maxt],lazy[maxt]; void build(const ll l,const ll r,const ll p) { if(l>r) return; if(l==r) {tree[p]=MU[l];return;} ll m=(l+r)>>,dp=p<<; build(l,m,dp);build(m+,r,dp+); tree[p]=tree[dp]+tree[dp+]; } inline void Free(cl l,cl r,cl p) { ll m=(l+r)>>,dp=p<<; tree[dp]+=(m-l+)*lazy[p];tree[dp+]+=(r-m)*lazy[p]; lazy[dp]+=lazy[p];lazy[dp+]+=lazy[p]; lazy[p]=; } inline )*v;lazy[p]+=v;} void add(cl l,cl r,cl p,cl aiml,cl aimr,cl v) { if(l>r) return; if(l>aimr||r<aiml) {return;} if(l>=aiml&&r<=aimr) {wohenlan(l,r,p,v);return;} Free(l,r,p); ll m=(l+r)>>,dp=p<<; add(l,m,dp,aiml,aimr,v);add(m+,r,dp+,aiml,aimr,v); tree[p]=tree[dp]+tree[dp+]; } ll ask(cl l,cl r,cl p,cl aiml,cl aimr) { ; ;} if(l>=aiml&&r<=aimr) {return tree[p];} Free(l,r,p); ll m=(l+r)>>,dp=p<<; ,r,dp+,aiml,aimr); } int main() { qr(n);qr(m); ;i<=n;++i) qr(MU[i]); build(1ll,n,1ll); while(m--) { sign=a=b=;qr(sign);qr(a);qr(b); ) { c=;qr(c); add(,n,,a,b,c); } , n, , a, b)); } ; }
Summary
1、线段树大小要开4*n。理论上线段树会有2*n个子节点,但是试试这棵线段树:1 2 3 4 5
如图所示:
可以看到,节点数确实是6*2-1=11个,但是由于我们每个节点编号都严格按照母节点*2(+1)进行编号,所以我们的编号开到了2*n之外。开4*n是比较保险的。
2、注意对lazy标记的free操作要在确定区间可以再分以后进行。即先写
if(l>=aiml&&r<=aimr) {wohenlan(l,r,p,v);return;}
或
if(l>=aiml&&r<=aimr) {return tree[p];}
后,如果没有return,则证明区间一定是可再分的,即还没有递归到叶节点,这时才可以进行free操作。否则的话考虑在叶节点的编号可能大于2*n,我们在叶节点free了一下,标记被下放到了4*n以外……
然后你就炸了。
【线段树】【P3372】模板-线段树的更多相关文章
- 【数据结构与算法】Trie(前缀树)模板和例题
Trie 树的模板 Trie 树的简介 Trie树,又称字典树,单词查找树或者前缀树,是一种用于快速检索的多叉树结构,如英文字母的字典树是一个26叉树,数字的字典树是一个10叉树.他的核心思想是空间换 ...
- hdu 1754 I Hate It (模板线段树)
http://acm.hdu.edu.cn/showproblem.php?pid=1754 I Hate It Time Limit: 9000/3000 MS (Java/Others) M ...
- hdu3966 树链剖分点权模板+线段树区间更新/树状数组区间更新单点查询
点权树的模板题,另外发现树状数组也是可以区间更新的.. 注意在对链进行操作时方向不要搞错 线段树版本 #include<bits/stdc++.h> using namespace std ...
- 洛谷P3373 [模板]线段树 2(区间增减.乘 区间求和)
To 洛谷.3373 [模板]线段树2 题目描述 如题,已知一个数列,你需要进行下面两种操作: 1.将某区间每一个数加上x 2.将某区间每一个数乘上x 3.求出某区间每一个数的和 输入输出格式 输入格 ...
- LuoguP3834 【模板】可持久化线段树 1(主席树)|| 离散化
题目:[模板]可持久化线段树 1(主席树) 不知道说啥. #include<cstdio> #include<cstring> #include<iostream> ...
- 【洛谷P3834】(模板)可持久化线段树 1(主席树)
[模板]可持久化线段树 1(主席树) https://www.luogu.org/problemnew/show/P3834 主席树支持历史查询,空间复杂度为O(nlogn),需要动态开点 本题用一个 ...
- Pascal 线段树 lazy-tag 模板
先说下我的代码风格(很丑,勿喷) maxn表示最大空间的四倍 tree数组表示求和的线段树 delta表示增减的增量标记 sign表示覆盖的标记 delta,sign实际上都是lazy标志 pushd ...
- 有趣的线段树模板合集(线段树,最短/长路,单调栈,线段树合并,线段树分裂,树上差分,Tarjan-LCA,势能线段树,李超线段树)
线段树分裂 以某个键值为中点将线段树分裂成左右两部分,应该类似Treap的分裂吧(我菜不会Treap).一般应用于区间排序. 方法很简单,就是把分裂之后的两棵树的重复的\(\log\)个节点新建出来, ...
- 洛谷P3834 [模板]可持久化线段树1(主席树) [主席树]
题目传送门 可持久化线段树1(主席树) 题目背景 这是个非常经典的主席树入门题——静态区间第K小 数据已经过加强,请使用主席树.同时请注意常数优化 题目描述 如题,给定N个正整数构成的序列,将对于指定 ...
随机推荐
- Git一分钟系列--快速安装git客户端
在项目开发过程中,几乎所有公司都会用到版本控制工具来管理自己的项目资源文件,比如Git,SVN. 什么是svn? 版本控制软件,通过svn来实现版本控制首先需要搭建一个服务器,在服务器上创建仓库保存项 ...
- .NET邮件发送详情
①百度下载个Free Smtp Server ②下载完成后,打开进去,点击左上角的options,弹出窗口,进行如下图选项即可(一般默认已正确选中),然后我们可以把这软件关了. (下次你重启电脑时,也 ...
- Python序列及其操作(常见)
python序列及函数入门认识: 0. 我们根据列表.元组和字符串的共同特点,把它们三统称为什么? 序列,因为他们有以下共同点: 1)都可以通过索引得到每一个元素 2)默认索引值总是从0开始(当 ...
- NGUI制作流光效果
效果展示: 技巧: 1.勾选UIPanel下的Normal启用UI的法线贴图,并建立带有法线贴图的UI对象(此处用NGUI自带的Reflector.Atlas中的图作为UI). 2.建立点光源并为其添 ...
- JS里点击事件判断是否 触发了节点 和给标签添加class属性
$("#activityType").click(function(e){ if(e.target==$("#bb")[0]){ var bb=document ...
- 【20180807模拟测试】T2 box
[问题描述] 有个桌子长 R 宽 C,被分为 R*C 个小方格.其中,一些方格上有箱子,一些方格上有按 钮,一些方格上有障碍物,一些方格上是空地.现在有个任务,需要把所有箱子推到这些按 钮上面.箱子有 ...
- lock+Condition
关键字 synchronized+wait/notify/notifyAll可以实现等待/通知模式,类ReentrantLock可以实现同样的功能,但需要借助Condition对象.Condition ...
- Python3 Tkinter-Radionbutton
1.创建单选按钮 from tkinter import * root=Tk() Radiobutton(root,text='b1').pack() Radiobutton(root,text='b ...
- vim编辑器配置及常用命令
最近工作不安分, 没有了刚入行时候的锐气, 不知道什么时候开始懈怠起来, 周末在电脑旁边看新闻, 搞笑图片, 追美剧, 一坐就是一天, 很是空虚. 我需要摆脱这种状态, 正好想学习一下安卓底层, An ...
- NSTimer使用注意事项
1.scheduled开头和非schedule的开头方法的区别.系统框架提供了几种创建NSTimer的方法,其中以scheduled开头的方法会自动把timer加入当前run loop,到了设定的时间 ...