[BZOJ4695]最假女选手:segment tree beats!
分析
segment tree beats!模板题。
看了gxz的博客突然发现自己写的mxbt和mnbt两个标记没用诶。
代码
#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!的更多相关文章
- bzoj4695 最假女选手
传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=4695 [题解] SegmentTree beats!(见jiry_2论文/营员交流) 考虑只 ...
- BZOJ4695 最假女选手(势能线段树)
BZOJ题目传送门 终于体会到初步掌握势能分析思想的重要性了. 一开始看题,感觉套路还是很一般啊qwq.直接在线段树上维护最大值和最小值,每次递归更新的时候,如果不能完全覆盖就暴力递归下去.挺好写的欸 ...
- 2018.07.27 bzoj4695: 最假女选手(线段树)
传送门 线段树好题 支持区间加,区间取min" role="presentation" style="position: relative;"> ...
- bzoj4695 最假女选手(势能线段树/吉司机线段树)题解
题意: 已知\(n\)个数字,进行以下操作: \(1.\)给一个区间\([L,R]\) 加上一个数\(x\) \(2.\)把一个区间\([L,R]\) 里小于\(x\) 的数变成\(x\) \(3.\ ...
- [学习笔记]Segment Tree Beats!九老师线段树
对于这样一类问题: 区间取min,区间求和. N<=100000 要求O(nlogn)级别的算法 直观体会一下,区间取min,还要维护区间和 增加的长度很不好求.... 然鹅, 从前有一个来自杭 ...
- BZOJ.4695.最假女选手(线段树 Segment tree Beats!)
题目链接 区间取\(\max,\ \min\)并维护区间和是普通线段树无法处理的. 对于操作二,维护区间最小值\(mn\).最小值个数\(t\).严格次小值\(se\). 当\(mn\geq x\)时 ...
- 【bzoj4695】最假女选手 线段树区间最值操作
题目描述 给定一个长度为 N 序列,编号从 1 到 N .要求支持下面几种操作:1.给一个区间[L,R] 加上一个数x 2.把一个区间[L,R] 里小于x 的数变成x 3.把一个区间[L,R] 里大于 ...
- (WC2016模拟十一)【BZOJ4695】最假女选手
ps:好久没更博啦……这几天连着有模拟赛,等初赛前后休息的时候来疯狂补坑吧……顺便补一下前面的数论啥的? 题解: mdzz我场上写了个15分暴力长度跟标算差不多... 线段树大法好啊!这题听说很多人做 ...
- Segment Tree Beats 区间最值问题
Segment Tree Beats 区间最值问题 线段树一类特殊技巧! 引出:CF671C Ultimate Weirdness of an Array 其实是考试题,改题的时候并不会区间取最值,区 ...
随机推荐
- Hadoop+HBase分布式部署
test 版本选择
- MyBatis一级缓存的笔记及记录
精髓内容来源于<图灵学院> 一.概述: 一级缓存是MyBatis天然自带的,是默认开启且没有关闭的地方,1级缓存只能作用于查询回话中,所以也叫会话缓存: 这里举个例子: 订单表存在一对多的 ...
- C语言 --- 函数指针(初级)
1.函数指针:指向函数的指针变量. 函数在内存中也是有地址的,函数名代表函数的内存地址. 例子:函数:int sum(int a,int b); int ...
- javaweb: request.getParameter()、request.setAttribute()与request.getAttribute()的作用 (转)
出处:https://blog.csdn.net/qq_41937388/article/details/87972914 1.request.getParameter()方法是获取通过类似post, ...
- 手把手教你vue配置请求本地json数据
本篇文章主要介绍了vue配置请求本地json数据的方法,分享给大家,具体如下:在build文件夹下找到webpack.dev.conf.js文件,在const portfinder = require ...
- Bootstrap中DropDown插件显示下拉列表,点击下拉列表区域,不会再自动关闭。
目标: Bootstrap中DropDown插件显示下拉列表,点击下拉列表区域,不会再自动关闭. 参考:http://v3.bootcss.com/javascript/#dropdowns / ...
- IA-32 Assembly Language Reference Manual
Load Full Pointer (lds,les, lfs, lgs, and lss) lds{wl} mem[32|48], reg[16|32]les{wl} mem[32|48], reg ...
- luogu P4631 [APIO2018] Circle selection 选圆圈
传送门 那个当前半径最大的圆可以用堆维护.这道题一个想法就是优化找和当前圆有交的圆的过程.考虑对于所有圆心建KD-tree,然后在树上遍历的找这样的点.只要某个点子树内的点构成的矩形区域到当前圆心的最 ...
- 吴恩达深度学习:2.16关于python/numpy
- webpack4导入jQuery的新方案
本文的目的 拒绝全局导入jQuery!! 拒绝script导入jQuery!! 找到一种只在当前js组件中引入jQuery,并且使用webpack切割打包的方案! 测试环境 以下测试在webpack3 ...