洛谷题解P4314CPU监控--线段树
题目链接
https://www.luogu.org/problemnew/show/P4314
https://www.lydsy.com/JudgeOnline/problem.php?id=3064
分析
其实我是在看吉司机线段树课件时看到这题很感兴趣就跑过来做
显然如果数据小一点可以用分块什么的比较好搞
但是这个数据范围可能用\(log N\)的数据结构更舒服一点
怎么搞呢?请阅读国家集训队2016论文集之《区间最值操作与历史最值问题——杭州学军中学 吉如一》,对,就是我们敬爱可亲的吉司机.
看不懂?实际上就是告诉我们维护6个\(lazy\)_\(tag\):
\(nmx\)表示当前区间最大值,\(add\)表示当前区间加法标记,\(set\)表示当前区间赋值标记
\(pmx\)表示当前区间历史最大值,\(padd\)表示当前区间在下传此标记前时历史最大加法标记,\(pset\)表示当前区间在下传此标记前历史最大赋值标记
这样\(lazy\)_\(tag\)之间的合并就比较显然了
然后再结合论文食用,或是看代码理解一下
当然GXZlegend大佬使用吉司机的另一个方法也是可行的
http://www.cnblogs.com/GXZlegend/p/8315275.html
注意
我查错又查了一个小时
注意不要把\(-inf\)写成\(inf\)
在\(pushdown\)时思维一定要清晰,注意是哪些标记会对其他标记产生影响
代码
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <algorithm>
#include <cctype>
#include <iostream>
#define ll long long
#define ri register int
using std::max;
const int inf=0x3f3f3f3f;
const int maxn=100005;
template <class T>inline void read(T &x){
x=0;int ne=0;char c;
while(!isdigit(c=getchar()))ne=c=='-';
x=c-48;
while(isdigit(c=getchar()))x=(x<<3)+(x<<1)+c-48;
x=ne?-x:x;return ;
}
int n;
int nmx[maxn<<2],add[maxn<<2],set[maxn<<2];
int pmx[maxn<<2],padd[maxn<<2],pset[maxn<<2];
int num[maxn];
int L,R,dta;
void build(int now,int l,int r){
set[now]=nmx[now]=pmx[now]=pset[now]=-inf;
padd[now]=add[now]=0;
if(l==r){
nmx[now]=pmx[now]=num[l];
return ;
}
int mid=(l+r)>>1;
build(now<<1,l,mid);
build(now<<1|1,mid+1,r);
nmx[now]=pmx[now]=max(nmx[now<<1],nmx[now<<1|1]);
return ;
}
inline void pushdown(int now){
if(padd[now]){
pmx[now<<1]=max(pmx[now<<1],nmx[now<<1]+padd[now]);
if(set[now<<1]!=-inf)
pset[now<<1]=max(pset[now<<1],set[now<<1]+padd[now]);
else
padd[now<<1]=max(padd[now<<1],add[now<<1]+padd[now]);
pmx[now<<1|1]=max(pmx[now<<1|1],nmx[now<<1|1]+padd[now]);
if(set[now<<1|1]!=-inf)
pset[now<<1|1]=max(pset[now<<1|1],set[now<<1|1]+padd[now]);
else
padd[now<<1|1]=max(padd[now<<1|1],add[now<<1|1]+padd[now]);
padd[now]=0;
}
if(pset[now]!=-inf){
pmx[now<<1]=max(pmx[now<<1],pset[now]);
pset[now<<1]=max(pset[now<<1],pset[now]);
pmx[now<<1|1]=max(pmx[now<<1|1],pset[now]);
pset[now<<1|1]=max(pset[now<<1|1],pset[now]);
pset[now]=-inf;
}
if(add[now]){
nmx[now<<1]+=add[now];
pmx[now<<1]=max(pmx[now<<1],nmx[now<<1]);
if(set[now<<1]!=-inf){
set[now<<1]+=add[now];
pset[now<<1]=max(pset[now<<1],set[now<<1]);
}
else {
add[now<<1]+=add[now];
padd[now<<1]=max(padd[now<<1],add[now<<1]);
}
nmx[now<<1|1]+=add[now];
pmx[now<<1|1]=max(pmx[now<<1|1],nmx[now<<1|1]);
if(set[now<<1|1]!=-inf){
set[now<<1|1]+=add[now];
pset[now<<1|1]=max(pset[now<<1|1],set[now<<1|1]);
}
else {
add[now<<1|1]+=add[now];
padd[now<<1|1]=max(padd[now<<1|1],add[now<<1|1]);
}
add[now]=0;
}
if(set[now]!=-inf){
nmx[now<<1]=set[now];
pmx[now<<1]=max(pmx[now<<1],nmx[now<<1]);
set[now<<1]=set[now];
pset[now<<1]=max(pset[now<<1],set[now<<1]);
nmx[now<<1|1]=set[now];
pmx[now<<1|1]=max(pmx[now<<1|1],nmx[now<<1|1]);
set[now<<1|1]=set[now];
pset[now<<1|1]=max(pset[now<<1|1],set[now<<1|1]);
set[now]=-inf;
add[now<<1]=add[now<<1|1]=0;
}
return ;
}
void update_add(int now,int l,int r){
if(L<=l&&r<=R){
nmx[now]+=dta;
pmx[now]=max(pmx[now],nmx[now]);
if(set[now]!=-inf){
set[now]+=dta;
pset[now]=max(pset[now],set[now]);
}
else{
add[now]+=dta;
padd[now]=max(padd[now],add[now]);
}
return ;
}
int mid=(l+r)>>1;
pushdown(now);
if(L<=mid)update_add(now<<1,l,mid);
if(mid<R)update_add(now<<1|1,mid+1,r);
nmx[now]=max(nmx[now<<1],nmx[now<<1|1]);
pmx[now]=max(pmx[now<<1],pmx[now<<1|1]);
return ;
}
void update_set(int now,int l,int r){
if(L<=l&&r<=R){
nmx[now]=dta;
pmx[now]=max(pmx[now],dta);
set[now]=dta;
pset[now]=max(pset[now],dta);
add[now]=0;
return ;
}
int mid=(l+r)>>1;
pushdown(now);
if(L<=mid)update_set(now<<1,l,mid);
if(mid<R)update_set(now<<1|1,mid+1,r);
nmx[now]=max(nmx[now<<1],nmx[now<<1|1]);
pmx[now]=max(pmx[now<<1],pmx[now<<1|1]);
return;
}
int query_now(int now,int l,int r){
if(L<=l&&r<=R){
return nmx[now];
}
int ans=-inf,mid=(l+r)>>1;
pushdown(now);
if(L<=mid)ans=max(ans,query_now(now<<1,l,mid));
if(mid<R)ans=max(ans,query_now(now<<1|1,mid+1,r));
nmx[now]=max(nmx[now<<1],nmx[now<<1|1]);
pmx[now]=max(pmx[now<<1],pmx[now<<1|1]);
return ans;
}
int query_history(int now,int l,int r){
if(L<=l&&r<=R){
return pmx[now];
}
int ans=-inf,mid=(l+r)>>1;
pushdown(now);
if(L<=mid)ans=max(ans,query_history(now<<1,l,mid));
if(mid<R)ans=max(ans,query_history(now<<1|1,mid+1,r));
nmx[now]=max(nmx[now<<1],nmx[now<<1|1]);
pmx[now]=max(pmx[now<<1],pmx[now<<1|1]);
return ans;
}
int q;
int main(){
int x,y,z;
char opt[5];
read(n);
for(ri i=1;i<=n;i++)read(num[i]);
build(1,1,n);
read(q);
while(q--){
scanf("%s",opt);
read(x),read(y);
L=x,R=y;
if(opt[0]=='Q'){
printf("%d\n",query_now(1,1,n));
}
else if(opt[0]=='A'){
printf("%d\n",query_history(1,1,n));
}
else if(opt[0]=='P'){
read(dta);
update_add(1,1,n);
}
else{
read(dta);
update_set(1,1,n);
}
}
return 0;
}
目前洛谷rank 2,在BZOJ上被吊打
洛谷题解P4314CPU监控--线段树的更多相关文章
- 题解——洛谷P2781 传教(线段树)
可以说是数据结构学傻了的典型案例了 昨天跳到这题上 然后思考了一下 噫!好!线段树裸题 然后打完板子,发现\( n \le 10^9 \) 显然线段树直接做不太行 然后这题又只有普及的难度 然后我就 ...
- 【BZOJ】1012: [JSOI2008]最大数maxnumber /【洛谷】1198(线段树)
Description 现在请求你维护一个数列,要求提供以下两种操作:1. 查询操作.语法:Q L 功能:查询当前数列中末尾L个数中的最大的数,并输出这个数的值.限制:L不超过当前数列的长度.2. 插 ...
- 洛谷P3372/poj3468(线段树lazy_tag)(询问区间和,支持区间修改)
洛谷P3372 //线段树 询问区间和,支持区间修改 #include <cstdio> using namespace std; struct treetype { int l,r; l ...
- 洛谷P4428二进制 [BJOI2018] 线段树
正解:线段树 解题报告: 传送门! 话说开始看到这题的时候我想得hin简单 因为关于%3有个性质就是说一个数的各个位数之和%3=这个数%3嘛,小学基础知识? 我就想着,就直接建一棵树,只是这棵树要用个 ...
- BZOJ2141&洛谷1975 排队 【线段树套treap】
题目 排排坐,吃果果,生果甜嗦嗦,大家笑呵呵.你一个,我一个,大的分给你,小的留给我,吃完果果唱支歌,大家乐和和. 红星幼儿园的小朋友们排起了长长地队伍,准备吃果果.不过因为小朋友们的身高有所区别,排 ...
- 洛谷P3707 [SDOI2017]相关分析(线段树)
题目描述 Frank对天文学非常感兴趣,他经常用望远镜看星星,同时记录下它们的信息,比如亮度.颜色等等,进而估算出星星的距离,半径等等. Frank不仅喜欢观测,还喜欢分析观测到的数据.他经常分析两个 ...
- 【洛谷P2894】Hotel 线段树+二分查询
题目大意:给定一个长度为 N 的序列,每个点有两种状态 1/0,表示占有和空闲,现支持 first-fit 查询是否有一段连续的长度为 X 的空闲子序列和区间赋值操作. 题解:get到了线段树新技能. ...
- 洛谷$P2486\ [SDOI2011]$染色 线段树+树链剖分
正解:线段树+树链剖分 解题报告: 传送门$QwQ$ 其实是道蛮板子的题,,,但因为我写得很呆然后写了贼久之后发现想法有问题要重构,就很难受,就先写个题解算了$kk$ 考虑先跑个树剖,然后按$dfn$ ...
- 洛谷$P1712\ [NOI2016]$区间 线段树
正解:线段树 解题报告: 传送门$QwQ$ $umm$很久以前做的了来补个题解$QwQ$ 考虑给每个区间按权值($r-l$从大往小排序,依次加入,然后考虑如果有一个位置被覆盖次数等于$m$了就可以把权 ...
随机推荐
- Thinkphp5 的sesssion在同一个控制器不同的方法无法获取session的原因和对策
这一段在用thinkPHP5开发微信小程序接口的时候,在同一个控制器一个方法中存入session,在另一个方法中取出session,一直都是无法取出. 查阅各种资料得到原因:thinkPHP5里面的s ...
- 异步发送表单数据到JavaBean,并响应JSON文本返回
1) 提交表单后,将JavaBean信息以JSON文本形式返回到浏览器 <form> 编号:<input type="text" name="id&q ...
- layui上传文件前加入确认提示
//上传文件 upload: function () { layui.use('upload', function () { var upload = layui.upload; //执行实例 var ...
- intellij import包 顺序调整
intellij中自动import的包顺序与eclipse不太一致,可以参照以下方式进行调整: eclipse中(笔者用的是eclipse luna)导入包的顺序依次是: javajavaxorgco ...
- Kafka API使用
- MATLAB学习(六)绘图图形功能
>> x=0:.1:2*pi;plot(x,sin(x),x,cos(x)) >> plot(x,sin(x),'p ...
- vagrant虚拟机共享目录在windows宿主下的禁忌
问题背景 宿主环境:Windows10 开发环境:vagrant(ubuntu) 操作目录:synced_folder (共享目录 ) 执行命令:npm install 错误信息: npm ERR! ...
- Git(4):远程仓库
添加\连接远程库 目前我们使用到的 Git 命令都是在本地执行,如果你想通过 Git 分享你的代码或者与其他开发人员合作. 你就需要将数据放到一台其他开发人员能够连接的服务器上. 远程仓库可以是Git ...
- JavaScript高程第三版笔记-DOM扩展
在那个刀耕火种的年代,用过jQuery的都体会到了jQuery带来的便捷,尤其是元素选择器. jQuery(www.jquery.com)的核心就是通过 CSS 选择符查询 DOM 文档取得元素的引用 ...
- kubernetes/dashboard Getting Started
Kubernetes Dashboard is a general purpose, web-based UI for Kubernetes clusters. It allows users to ...