分析

segment tree beats!模板题。

看了gxz的博客突然发现自己写的mxbtmnbt两个标记没用诶。

代码

#include <bits/stdc++.h>
#define rin(i,a,b) for(register int i=(a);i<=(b);++i)
#define irin(i,a,b) for(register int i=(a);i>=(b);--i)
#define trav(i,a) for(register int i=head[a];i;i=e[i].nxt)
typedef long long LL;
using std::cin;
using std::cout;
using std::endl; inline int read(){
int x=0,f=1;char ch=getchar();
while(!isdigit(ch)){if(ch=='-')f=-1;ch=getchar();}
while(isdigit(ch)){x=x*10+ch-'0';ch=getchar();}
return x*f;
} const int MAXN=5e5+5; int n,m,a[MAXN];
int maxn[MAXN<<2],minn[MAXN<<2],
mxcnt[MAXN<<2],mncnt[MAXN<<2],
mxsec[MAXN<<2],mnsec[MAXN<<2],
tag[MAXN<<2],
ql,qr,kk;
LL sum[MAXN<<2];
bool mxbt[MAXN<<2],mnbt[MAXN<<2]; #define mid ((l+r)>>1)
#define lc (o<<1)
#define rc ((o<<1)|1) inline void pushup(int o){
sum[o]=sum[lc]+sum[rc];
if(maxn[lc]>maxn[rc]){
maxn[o]=maxn[lc];
mxcnt[o]=mxcnt[lc];
mxsec[o]=std::max(mxsec[lc],maxn[rc]);
}
else if(maxn[lc]<maxn[rc]){
maxn[o]=maxn[rc];
mxcnt[o]=mxcnt[rc];
mxsec[o]=std::max(mxsec[rc],maxn[lc]);
}
else{
maxn[o]=maxn[lc];
mxcnt[o]=mxcnt[lc]+mxcnt[rc];
mxsec[o]=std::max(mxsec[lc],mxsec[rc]);
}
if(minn[lc]<minn[rc]){
minn[o]=minn[lc];
mncnt[o]=mncnt[lc];
mnsec[o]=std::min(mnsec[lc],minn[rc]);
}
else if(minn[lc]>minn[rc]){
minn[o]=minn[rc];
mncnt[o]=mncnt[rc];
mnsec[o]=std::min(mnsec[rc],minn[lc]);
}
else{
minn[o]=minn[lc];
mncnt[o]=mncnt[lc]+mncnt[rc];
mnsec[o]=std::min(mnsec[lc],mnsec[rc]);
}
} inline void pushtag(int o,int l,int r,int _kk){
sum[o]+=1ll*_kk*(r-l+1);
maxn[o]+=_kk;
minn[o]+=_kk;
mxsec[o]+=_kk;
mnsec[o]+=_kk;
tag[o]+=_kk;
} inline void pushmaxbeat(int o,int _kk){
if(minn[o]>=_kk) return;
else if(minn[o]<_kk){
sum[o]+=1ll*(_kk-minn[o])*mncnt[o];
if(maxn[o]==minn[o]) maxn[o]=_kk;
else if(mxsec[o]==minn[o]) mxsec[o]=_kk;
minn[o]=_kk,mxbt[o]=true;
}
} inline void pushminbeat(int o,int _kk){
if(maxn[o]<=_kk) return;
else if(maxn[o]>_kk){
sum[o]+=1ll*(_kk-maxn[o])*mxcnt[o];
if(minn[o]==maxn[o]) minn[o]=_kk;
else if(mnsec[o]==maxn[o]) mnsec[o]=_kk;
maxn[o]=_kk,mnbt[o]=true;
}
} inline void pushdown(int o,int l,int r){
if(mxbt[o]){
pushmaxbeat(lc,minn[o]-tag[o]);
pushmaxbeat(rc,minn[o]-tag[o]);
mxbt[o]=false;
}
if(mnbt[o]){
pushminbeat(lc,maxn[o]-tag[o]);
pushminbeat(rc,maxn[o]-tag[o]);
mnbt[o]=false;
}
if(tag[o]){
pushtag(lc,l,mid,tag[o]);
pushtag(rc,mid+1,r,tag[o]);
tag[o]=0;
}
} void build(int o,int l,int r){
if(l==r){
maxn[o]=minn[o]=sum[o]=a[l];
mxcnt[o]=mncnt[o]=1;
mxsec[o]=-1e9,mnsec[o]=1e9;
return;
}
build(lc,l,mid);
build(rc,mid+1,r);
pushup(o);
} void add(int o,int l,int r){
if(ql<=l&&r<=qr){
pushtag(o,l,r,kk);
return;
}
pushdown(o,l,r);
if(mid>=ql) add(lc,l,mid);
if(mid<qr) add(rc,mid+1,r);
pushup(o);
} void checkmax(int o,int l,int r){
if(ql<=l&&r<=qr){
if(kk<=minn[o]) return;
else if(minn[o]<kk&&kk<mnsec[o]){
sum[o]+=1ll*(kk-minn[o])*mncnt[o];
if(maxn[o]==minn[o]) maxn[o]=kk;
else if(mxsec[o]==minn[o]) mxsec[o]=kk;
minn[o]=kk,mxbt[o]=true;
return;
}
}
pushdown(o,l,r);
if(mid>=ql) checkmax(lc,l,mid);
if(mid<qr) checkmax(rc,mid+1,r);
pushup(o);
} void checkmin(int o,int l,int r){
if(ql<=l&&r<=qr){
if(kk>=maxn[o]) return;
else if(maxn[o]>kk&&kk>mxsec[o]){
sum[o]+=1ll*(kk-maxn[o])*mxcnt[o];
if(minn[o]==maxn[o]) minn[o]=kk;
else if(mnsec[o]==maxn[o]) mnsec[o]=kk;
maxn[o]=kk,mnbt[o]=true;
return;
}
}
pushdown(o,l,r);
if(mid>=ql) checkmin(lc,l,mid);
if(mid<qr) checkmin(rc,mid+1,r);
pushup(o);
} LL querysum(int o,int l,int r){
if(ql<=l&&r<=qr) return sum[o];
pushdown(o,l,r);
LL ret=0;
if(mid>=ql) ret+=querysum(lc,l,mid);
if(mid<qr) ret+=querysum(rc,mid+1,r);
return ret;
} int querymax(int o,int l,int r){
if(ql<=l&&r<=qr) return maxn[o];
pushdown(o,l,r);
int ret=-1e9;
if(mid>=ql) ret=std::max(ret,querymax(lc,l,mid));
if(mid<qr) ret=std::max(ret,querymax(rc,mid+1,r));
return ret;
} int querymin(int o,int l,int r){
if(ql<=l&&r<=qr) return minn[o];
pushdown(o,l,r);
int ret=1e9;
if(mid>=ql) ret=std::min(ret,querymin(lc,l,mid));
if(mid<qr) ret=std::min(ret,querymin(rc,mid+1,r));
return ret;
} #undef mid
#undef lc
#undef rc int main(){
n=read();
rin(i,1,n) a[i]=read();
build(1,1,n);
m=read();
while(m--){
int opt=read(),l=read(),r=read();
if(opt==1){
int x=read();
ql=l,qr=r,kk=x;
add(1,1,n);
}
else if(opt==2){
int x=read();
ql=l,qr=r,kk=x;
checkmax(1,1,n);
}
else if(opt==3){
int x=read();
ql=l,qr=r,kk=x;
checkmin(1,1,n);
}
else if(opt==4){
ql=l,qr=r;
printf("%lld\n",querysum(1,1,n));
}
else if(opt==5){
ql=l,qr=r;
printf("%d\n",querymax(1,1,n));
}
else{
ql=l,qr=r;
printf("%d\n",querymin(1,1,n));
}
}
return 0;
}

[BZOJ4695]最假女选手:segment tree beats!的更多相关文章

  1. bzoj4695 最假女选手

    传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=4695 [题解] SegmentTree beats!(见jiry_2论文/营员交流) 考虑只 ...

  2. BZOJ4695 最假女选手(势能线段树)

    BZOJ题目传送门 终于体会到初步掌握势能分析思想的重要性了. 一开始看题,感觉套路还是很一般啊qwq.直接在线段树上维护最大值和最小值,每次递归更新的时候,如果不能完全覆盖就暴力递归下去.挺好写的欸 ...

  3. 2018.07.27 bzoj4695: 最假女选手(线段树)

    传送门 线段树好题 支持区间加,区间取min" role="presentation" style="position: relative;"> ...

  4. bzoj4695 最假女选手(势能线段树/吉司机线段树)题解

    题意: 已知\(n\)个数字,进行以下操作: \(1.\)给一个区间\([L,R]\) 加上一个数\(x\) \(2.\)把一个区间\([L,R]\) 里小于\(x\) 的数变成\(x\) \(3.\ ...

  5. [学习笔记]Segment Tree Beats!九老师线段树

    对于这样一类问题: 区间取min,区间求和. N<=100000 要求O(nlogn)级别的算法 直观体会一下,区间取min,还要维护区间和 增加的长度很不好求.... 然鹅, 从前有一个来自杭 ...

  6. BZOJ.4695.最假女选手(线段树 Segment tree Beats!)

    题目链接 区间取\(\max,\ \min\)并维护区间和是普通线段树无法处理的. 对于操作二,维护区间最小值\(mn\).最小值个数\(t\).严格次小值\(se\). 当\(mn\geq x\)时 ...

  7. 【bzoj4695】最假女选手 线段树区间最值操作

    题目描述 给定一个长度为 N 序列,编号从 1 到 N .要求支持下面几种操作:1.给一个区间[L,R] 加上一个数x 2.把一个区间[L,R] 里小于x 的数变成x 3.把一个区间[L,R] 里大于 ...

  8. (WC2016模拟十一)【BZOJ4695】最假女选手

    ps:好久没更博啦……这几天连着有模拟赛,等初赛前后休息的时候来疯狂补坑吧……顺便补一下前面的数论啥的? 题解: mdzz我场上写了个15分暴力长度跟标算差不多... 线段树大法好啊!这题听说很多人做 ...

  9. Segment Tree Beats 区间最值问题

    Segment Tree Beats 区间最值问题 线段树一类特殊技巧! 引出:CF671C Ultimate Weirdness of an Array 其实是考试题,改题的时候并不会区间取最值,区 ...

随机推荐

  1. Fescar使用(资料)

    fescar源码走读1:业务调用方 https://zhuanlan.zhihu.com/p/54659540   fescar源码走读2:fescar服务端 https://zhuanlan.zhi ...

  2. springboot - 应用实践(3)springboot的核心

    1.springboot的启动类与核心注解@SpringBootApplication 2.springboot基本配置 3.springboot自动配置原理

  3. Python 集合的各种操作 数学运算 关系操作 增删改查 生成式

    # 集合是无序的 是可变的 不能重复 允许数学运算 分散存储 # 创建# collegel = {'哲学','经济学','法学','教育学'}## # {'教育学', '经济学', '法学', '哲学 ...

  4. django-restframework使用

    安装restframework: pip install djangorestframework 修改项目settings.py: INSTALLED_APPS = [ 'django.contrib ...

  5. 786B - Legacy(线段树 + 最短路)线段树优化建图

    题意: 就是给定一张n nn个点的图,求源点s ss到每个点的单源最短路.这张图共有q组边,连边方式有3种: a→b ,边权为w的单向边:a→[l,r] ,即a到连续区间[l,r]中的每一个点都有一条 ...

  6. MyBatis的Insert操作详解

    一.前言 数据库操作怎能少了INSERT操作呢?下面记录MyBatis关于INSERT操作的笔记,以便日后查阅. 二. insert元素 属性详解 其属性如下: parameterType ,入参的全 ...

  7. jstat命令使用

    jstat命令使用 jstat是JDK自带的一个轻量级小工具,全称"Java Virtual Machine statistics monitoring tool",它位于java ...

  8. [PyQt5]动态显示matplotlib作图(一)

    完整实例 import sys from PyQt5.QtWidgets import QApplication, QMainWindow, QMenu, QVBoxLayout, QSizePoli ...

  9. sed 的用法

    sed 的语法: sed `样式命令` 文件 意思是:如果文件中的某一行符合”样式”,就执行指定的 sed 命令,如删除(d)或替换(s) “样式”使用一对//含括,表示寻找的意思.也可以指定数据行的 ...

  10. git 命令解析

    git 补丁 Git 提供了两种补丁方案:   (1)用 git diff 生成的UNIX标准补丁.diff文件:.diff文件只是记录文件改变的内容,不带有commit记录信息,多个commit可以 ...