题目链接

https://www.luogu.org/problemnew/show/P4314

https://www.lydsy.com/JudgeOnline/problem.php?id=3064

分析

其实我是在看吉司机线段树课件时看到这题很感兴趣就跑过来做

显然如果数据小一点可以用分块什么的比较好搞

但是这个数据范围可能用\(log N\)的数据结构更舒服一点

怎么搞呢?请阅读国家集训队2016论文集之《区间最值操作与历史最值问题——杭州学军中学 吉如一》,对,就是我们敬爱可亲的吉司机.

看不懂?实际上就是告诉我们维护6个\(lazy\)_\(tag\):

  1. \(nmx\)表示当前区间最大值,\(add\)表示当前区间加法标记,\(set\)表示当前区间赋值标记

  2. \(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监控--线段树的更多相关文章

  1. 题解——洛谷P2781 传教(线段树)

    可以说是数据结构学傻了的典型案例了 昨天跳到这题上 然后思考了一下 噫!好!线段树裸题 然后打完板子,发现\(  n \le 10^9 \) 显然线段树直接做不太行 然后这题又只有普及的难度 然后我就 ...

  2. 【BZOJ】1012: [JSOI2008]最大数maxnumber /【洛谷】1198(线段树)

    Description 现在请求你维护一个数列,要求提供以下两种操作:1. 查询操作.语法:Q L 功能:查询当前数列中末尾L个数中的最大的数,并输出这个数的值.限制:L不超过当前数列的长度.2. 插 ...

  3. 洛谷P3372/poj3468(线段树lazy_tag)(询问区间和,支持区间修改)

    洛谷P3372 //线段树 询问区间和,支持区间修改 #include <cstdio> using namespace std; struct treetype { int l,r; l ...

  4. 洛谷P4428二进制 [BJOI2018] 线段树

    正解:线段树 解题报告: 传送门! 话说开始看到这题的时候我想得hin简单 因为关于%3有个性质就是说一个数的各个位数之和%3=这个数%3嘛,小学基础知识? 我就想着,就直接建一棵树,只是这棵树要用个 ...

  5. BZOJ2141&洛谷1975 排队 【线段树套treap】

    题目 排排坐,吃果果,生果甜嗦嗦,大家笑呵呵.你一个,我一个,大的分给你,小的留给我,吃完果果唱支歌,大家乐和和. 红星幼儿园的小朋友们排起了长长地队伍,准备吃果果.不过因为小朋友们的身高有所区别,排 ...

  6. 洛谷P3707 [SDOI2017]相关分析(线段树)

    题目描述 Frank对天文学非常感兴趣,他经常用望远镜看星星,同时记录下它们的信息,比如亮度.颜色等等,进而估算出星星的距离,半径等等. Frank不仅喜欢观测,还喜欢分析观测到的数据.他经常分析两个 ...

  7. 【洛谷P2894】Hotel 线段树+二分查询

    题目大意:给定一个长度为 N 的序列,每个点有两种状态 1/0,表示占有和空闲,现支持 first-fit 查询是否有一段连续的长度为 X 的空闲子序列和区间赋值操作. 题解:get到了线段树新技能. ...

  8. 洛谷$P2486\ [SDOI2011]$染色 线段树+树链剖分

    正解:线段树+树链剖分 解题报告: 传送门$QwQ$ 其实是道蛮板子的题,,,但因为我写得很呆然后写了贼久之后发现想法有问题要重构,就很难受,就先写个题解算了$kk$ 考虑先跑个树剖,然后按$dfn$ ...

  9. 洛谷$P1712\ [NOI2016]$区间 线段树

    正解:线段树 解题报告: 传送门$QwQ$ $umm$很久以前做的了来补个题解$QwQ$ 考虑给每个区间按权值($r-l$从大往小排序,依次加入,然后考虑如果有一个位置被覆盖次数等于$m$了就可以把权 ...

随机推荐

  1. VS2015编译cef3-2357

    1.会遇到把警告当错误的情况 按照如下设置即可 2. 错误 C2334 “:”的前面有意外标记:跳过明显的函数体 (编译源文件 E:\cef_binary_3.2357.1271.g8e0674e_w ...

  2. 配置默认编码为utf8

    修改/etc/my.cnf配置文件,在[mysqld]下添加编码配置,如下所示: [mysqld] character_set_server=utf8 init_connect='SET NAMES ...

  3. GC垃圾回收 | 深入理解G1垃圾收集器和GC日志

    来源:并发编程网链接:http://ifeve.com/深入理解G1垃圾收集器/ G1 GC是Jdk7的新特性之一.Jdk7+版本都可以自主配置G1作为JVM GC选项:作为JVM GC算法的一次重大 ...

  4. buildscript和allprojects的作用和区别是什么?

    在Android Studio的Project的build.gradle中, // Top-level build file where you can add configuration optio ...

  5. java序列化与反序列化操作redis

     笔者在使用SSM框架项目部分功能进行测试需要使用到对象的序列化与反序列化 第一种方式:jackson Demo package com.dznfit.service; import com.dznf ...

  6. centos虚拟机存储扩容

    在vSphere Web Client上面创建的虚拟机,用了一段时间后存储无法满足需求,需要将原来的存储300G扩容到500G 点此编辑即可修改磁盘2的储存大小,但是修改此配置后,虚拟机centos是 ...

  7. SSIS数据同步实践

    SSIS数据同步实践   背景 在已初步验证不同实例下同构表数据同步方案之后,为了实现数据持续同步,需使用SSIS把之前的生成脚本和执行脚本的两个步骤组合在一起部署成包之后,通过JOB定时去执行: 测 ...

  8. 在DELPHI中动态创建控件以及控件的事件

    在DELPHI中我们经常要动态的创建控件以及控件的事件.例如,我们可能想根据程序需要动态的创建一些Tshape组件来创建某个图形,并使得在鼠标移动上去之后可以完成某些操作.这一般需要需要三步: 生成一 ...

  9. 安装horizon

    在控制节点上安装 controllerHost='controller' ADMIN_PASSWD='Ideal123!' 1.安装dashboard组件 yum -y install opensta ...

  10. Go语言中切片的内部实现和基础功能

    切片是一种数据结构,这种数据结构便于使用和管理数据集合.切片是围绕动态数组的概念构建的,可以按需自动增长和缩小.切片的动态增长是通过内置函数append来实现的.这个函数可以快速且高效的增长切片.还可 ...