BZOJ4695:最假女选手
浅谈区间最值操作和历史最值问题:https://www.cnblogs.com/AKMer/p/10225100.html
题目传送门:https://lydsy.com/JudgeOnline/problem.php?id=4695
吉司机线段树板子大集合。所有信息都封装在一个结构体里会比开多个数组快14秒。
注意暴力\(dfs\)子树时要\(pushdown\)。
时间复杂度:\(O(nlog^2n)\)
空间复杂度:\(O(n)\)
代码如下:
#include <cstdio>
#include <algorithm>
using namespace std;
typedef long long ll;
const int maxn=5e5+5,inf=1e9;
int n,m;
int a[maxn];
inline int read() {
int x=0,f=1;char ch=getchar();
for(;ch<'0'||ch>'9';ch=getchar())if(ch=='-')f=-1;
for(;ch>='0'&&ch<='9';ch=getchar())x=x*10+ch-'0';
return x*f;
}
struct segment_tree {
struct tree_node {
ll sum;
int cntA,cntZ,tag,A,B,Y,Z;//A最大值,B严格次大值,Y严格次小值,Z最小值
}tree[maxn<<2];
inline void update(int p) {
tree[p].A=max(tree[p<<1].A,tree[p<<1|1].A);
tree[p].Z=min(tree[p<<1].Z,tree[p<<1|1].Z);
tree[p].sum=tree[p<<1].sum+tree[p<<1|1].sum;
tree[p].cntA=(tree[p<<1].A==tree[p].A)*tree[p<<1].cntA;
tree[p].cntA+=(tree[p<<1|1].A==tree[p].A)*tree[p<<1|1].cntA;
tree[p].cntZ=(tree[p<<1].Z==tree[p].Z)*tree[p<<1].cntZ;
tree[p].cntZ+=(tree[p<<1|1].Z==tree[p].Z)*tree[p<<1|1].cntZ;
tree[p].B=tree[p<<1|1].A!=tree[p].A?tree[p<<1|1].A:tree[p<<1|1].B;
tree[p].B=max(tree[p].B,tree[p<<1].A!=tree[p].A?tree[p<<1].A:tree[p<<1].B);
tree[p].Y=tree[p<<1|1].Z!=tree[p].Z?tree[p<<1|1].Z:tree[p<<1|1].Y;
tree[p].Y=min(tree[p].Y,tree[p<<1].Z!=tree[p].Z?tree[p<<1].Z:tree[p<<1].Y);
}
inline void build(int p,int l,int r) {
if(l==r) {
tree[p].cntA=tree[p].cntZ=1;
tree[p].sum=tree[p].A=tree[p].Z=a[l];
tree[p].B=-inf,tree[p].Y=inf;
return;
}
int mid=(l+r)>>1;
build(p<<1,l,mid);
build(p<<1|1,mid+1,r);
update(p);
}
inline void add_tag(int p,int l,int r,int v) {
tree[p].A+=v,tree[p].Z+=v,tree[p].tag+=v;
if(tree[p].Y!=inf)tree[p].Y+=v;
if(tree[p].B!=-inf)tree[p].B+=v;
tree[p].sum+=1ll*(r-l+1)*v;
}
inline void max_tag(int p,int v) {
tree[p].sum+=1ll*tree[p].cntZ*(v-tree[p].Z),tree[p].Z=v;
if(tree[p].Y==inf)tree[p].A=v;
else tree[p].B=max(tree[p].B,v);
}
inline void min_tag(int p,int v) {
tree[p].sum-=1ll*tree[p].cntA*(tree[p].A-v),tree[p].A=v;
if(tree[p].B==-inf)tree[p].Z=v;
else tree[p].Y=min(tree[p].Y,v);
}
inline void solveMax(int p,int l,int r,int limit) {
if(limit<=tree[p].Z)return;
if(limit>tree[p].Z&&limit<tree[p].Y) {
max_tag(p,limit);
return;
}
int mid=(l+r)>>1;push_down(p,l,r);
solveMax(p<<1,l,mid,limit);
solveMax(p<<1|1,mid+1,r,limit);
update(p);
}
inline void solveMin(int p,int l,int r,int limit) {
if(limit>=tree[p].A)return;
if(limit<tree[p].A&&limit>tree[p].B) {
min_tag(p,limit);
return;
}
int mid=(l+r)>>1;push_down(p,l,r);
solveMin(p<<1,l,mid,limit);
solveMin(p<<1|1,mid+1,r,limit);
update(p);
}
inline void push_down(int p,int l,int r) {
int mid=(l+r)>>1;
if(tree[p].tag) {
add_tag(p<<1,l,mid,tree[p].tag);
add_tag(p<<1|1,mid+1,r,tree[p].tag);
tree[p].tag=0;
}
solveMin(p<<1,l,mid,tree[p].A);
solveMin(p<<1|1,mid+1,r,tree[p].A);
solveMax(p<<1,l,mid,tree[p].Z);
solveMax(p<<1|1,mid+1,r,tree[p].Z);
}
inline void ADD(int p,int l,int r,int L,int R,int v) {
if(L<=l&&r<=R) {
add_tag(p,l,r,v);
return;
}
int mid=(l+r)>>1;push_down(p,l,r);
if(L<=mid)ADD(p<<1,l,mid,L,R,v);
if(R>mid)ADD(p<<1|1,mid+1,r,L,R,v);
update(p);
}
inline void MAX(int p,int l,int r,int L,int R,int v) {
if(tree[p].Z>=v)return;
if(L<=l&&r<=R) {
solveMax(p,l,r,v);
return;
}
int mid=(l+r)>>1;push_down(p,l,r);
if(L<=mid)MAX(p<<1,l,mid,L,R,v);
if(R>mid)MAX(p<<1|1,mid+1,r,L,R,v);
update(p);
}
inline void MIN(int p,int l,int r,int L,int R,int v) {
if(tree[p].A<=v)return;
if(L<=l&&r<=R) {
solveMin(p,l,r,v);
return;
}
int mid=(l+r)>>1;push_down(p,l,r);
if(L<=mid)MIN(p<<1,l,mid,L,R,v);
if(R>mid)MIN(p<<1|1,mid+1,r,L,R,v);
update(p);
}
inline ll querySum(int p,int l,int r,int L,int R) {
if(L<=l&&r<=R)return tree[p].sum;
int mid=(l+r)>>1;ll res=0;push_down(p,l,r);
if(L<=mid)res=querySum(p<<1,l,mid,L,R);
if(R>mid)res+=querySum(p<<1|1,mid+1,r,L,R);
return res;
}
inline int queryMax(int p,int l,int r,int L,int R) {
if(L<=l&&r<=R)return tree[p].A;
int mid=(l+r)>>1,res=-inf;push_down(p,l,r);
if(L<=mid)res=max(res,queryMax(p<<1,l,mid,L,R));
if(R>mid)res=max(res,queryMax(p<<1|1,mid+1,r,L,R));
return res;
}
inline int queryMin(int p,int l,int r,int L,int R) {
if(L<=l&&r<=R)return tree[p].Z;
int mid=(l+r)>>1,res=inf;push_down(p,l,r);
if(L<=mid)res=min(res,queryMin(p<<1,l,mid,L,R));
if(R>mid)res=min(res,queryMin(p<<1|1,mid+1,r,L,R));
return res;
}
}T;
int main() {
n=read();
for(int i=1;i<=n;i++)
a[i]=read();
T.build(1,1,n);
m=read();
for(int i=1;i<=m;i++) {
int opt=read(),l=read(),r=read(),v=(opt<4?read():0);
if(opt==1)T.ADD(1,1,n,l,r,v);
if(opt==2)T.MAX(1,1,n,l,r,v);
if(opt==3)T.MIN(1,1,n,l,r,v);
if(opt==4)printf("%lld\n",T.querySum(1,1,n,l,r));
if(opt==5)printf("%d\n",T.queryMax(1,1,n,l,r));
if(opt==6)printf("%d\n",T.queryMin(1,1,n,l,r));
}
return 0;
}
BZOJ4695:最假女选手的更多相关文章
- BZOJ4695 最假女选手(势能线段树)
BZOJ题目传送门 终于体会到初步掌握势能分析思想的重要性了. 一开始看题,感觉套路还是很一般啊qwq.直接在线段树上维护最大值和最小值,每次递归更新的时候,如果不能完全覆盖就暴力递归下去.挺好写的欸 ...
- 2018.07.27 bzoj4695: 最假女选手(线段树)
传送门 线段树好题 支持区间加,区间取min" role="presentation" style="position: relative;"> ...
- bzoj4695 最假女选手
传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=4695 [题解] SegmentTree beats!(见jiry_2论文/营员交流) 考虑只 ...
- [BZOJ4695]最假女选手:segment tree beats!
分析 segment tree beats!模板题. 看了gxz的博客突然发现自己写的mxbt和mnbt两个标记没用诶. 代码 #include <bits/stdc++.h> #defi ...
- bzoj4695 最假女选手(势能线段树/吉司机线段树)题解
题意: 已知\(n\)个数字,进行以下操作: \(1.\)给一个区间\([L,R]\) 加上一个数\(x\) \(2.\)把一个区间\([L,R]\) 里小于\(x\) 的数变成\(x\) \(3.\ ...
- bzoj 4695 最假女选手 吉利线段树
最假女选手 Time Limit: 50 Sec Memory Limit: 128 MBSubmit: 480 Solved: 118[Submit][Status][Discuss] Desc ...
- 【bzoj4695】最假女选手 线段树区间最值操作
题目描述 给定一个长度为 N 序列,编号从 1 到 N .要求支持下面几种操作:1.给一个区间[L,R] 加上一个数x 2.把一个区间[L,R] 里小于x 的数变成x 3.把一个区间[L,R] 里大于 ...
- 【bzoj4695】最假女选手
zcy的励志故事.jpg 傻逼zcy突然想立一个flag,写一个segment-tree-beats的题娱乐一下 于是他就想起了这道题. 他打算今晚写完 然后光是写他就写的头昏脑涨,还犯了询问写反这种 ...
- (WC2016模拟十一)【BZOJ4695】最假女选手
ps:好久没更博啦……这几天连着有模拟赛,等初赛前后休息的时候来疯狂补坑吧……顺便补一下前面的数论啥的? 题解: mdzz我场上写了个15分暴力长度跟标算差不多... 线段树大法好啊!这题听说很多人做 ...
随机推荐
- 【CodeM初赛A轮】D 分解质因数+暴力
题目描述树链是指树里的一条路径.美团外卖的形象代言人袋鼠先生最近在研究一个特殊的最长树链问题.现在树中的每个点都有一个正整数值,他想在树中找出最长的树链,使得这条树链上所有对应点的值的最大公约数大于1 ...
- 我的Java开发学习之旅------>Base64的编码思想以及Java实现
Base64是一种用64个字符来表示任意二进制数据的方法. 用记事本打开exe.jpg.pdf这些文件时,我们都会看到一大堆乱码,因为二进制文件包含很多无法显示和打印的字符,所以,如果要让记事本这样的 ...
- 教你使用SQL查询(1-12)
教你使用 Select 查询语句 (1) SELECT 语句基本语法简介 http://jimshu.blog.51cto.com/3171847/1363101(2) TOP 和 OFFSET 筛选 ...
- Java并发之LinkedBlockingQueue
上一篇我们已经学习过了 ArrayBlockingQueue的知识及相关方法的使用,这一篇我们就来再学习一下ArrayBlockingQueue的亲戚 LinkedBlockingQueue.在集合类 ...
- pinpoint本地开发——collector
本地启动collector 启动前准备 启动之前,要先确保本地已经可以正常package,install 必须保证install成功,才能进行后续步骤,无法install或者package参考[pin ...
- J.U.C重入锁
ReentrantLock重入锁 ReentrantLock是Java并发包中互斥锁,它有公平锁和非公平锁两种实现方式, 重入的意思就是,如果已经获得了锁,如果执行期间还需要获得这个锁的话,会直接获得 ...
- iOS UIImage UIImageView 展示图片 不变形 处理
展示图片 时候 固定 了 imageView 的大小 图片 也裁剪了 尽量保持比例 可是 还是失真 变形了 这张图 ui 要求展示的UIimageView 大小 是固定 的 ,传过来的 图片 是 ...
- Android Media应用开发
Java层API : AudioRecord 采集音频(PCM格式)AudioTrack 播放音频 Camera 用来采集摄像头的数据Surface / SurfaceView / GLSurface ...
- P3988 [SHOI2013]发牌
题目 P3988 [SHOI2013]发牌 做法 我们切牌时的状态: 手玩几次后我们发现切\(K\)次牌就是求堆顶一下的\(K+1\)大值,套上主席树就好了 My complete code #inc ...
- Referrer-Policy常见属性
Referrer-Policy(来源协议)用来规定什么情况下显示Referer字段及refer字段内显示多少信息. 备注: referer实际上是对referrer的误写,因为写错的人多了也就正确了. ...