[LOJ2736] [JOISC 2016 Day 3] 回转寿司 (分块+堆)

题面

给出一个有n 个点的环,环上各点有一个初始权值 \(a_i\)

给出 Q 个询问,每次询问给出一个区间 [l,r]和一个值 A,对于 A 的变动定义如下

for (int i = l; i <= r; i++) if(a[i] > A) swap(a[i],A);

对于每个询问,回答遍历完区间[l,r]后 A的最终值。

分析

这种交换看起来很难用一般的数据结构维护,考虑对序列分块。

首先,我们发现如果一个块里面所有的数都<A,那么什么变化都不会发生。否则每个块中最大的数会被A替换掉,A变成\(max(A,max(a_i)) (i\in [l,r])\)。因此对每个块内的数维护一个大根堆,只有堆顶比当前A小的时候才会整块修改。

一般分块的套路,整块修改的时候只需要打一个标记,两端的小块直接按题面模拟遍历即可。遍历两端的时候要把标记下推,同时重构两端的块

考虑标记下推,我们发现对于两个不同的标记Ax,Ay,若Ax<Ay,最大的数会被替换成Ax。所以我们每个块用一个小根堆存储标记,遍历重构的块的时候若a[i]>A,把原来的a[i]插入小根堆,然后swap(A,a[i])。可以证明,这和多次遍历块来交换每个标记是一样的。

代码

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<queue>
#define maxn 400000
#define maxb 1000
using namespace std;
inline void qread(int &x) {
x=0;
int sign=1;
char c=getchar();
while(c<'0'||c>'9') {
if(c=='-') sign=-1;
c=getchar();
}
while(c>='0'&&c<='9') {
x=x*10+c-'0';
c=getchar();
}
x=x*sign;
}
inline void qprint(int x) {
if(x<0) {
putchar('-');
qprint(-x);
} else if(x==0) {
putchar('0');
return;
} else {
if(x>=10) qprint(x/10);
putchar('0'+x%10);
}
} int n,m;
int a[maxn+5]; int bsz;
int bel[maxn+5];
inline int lb(int id){
return bsz*(id-1)+1;
}
inline int rb(int id){
return bsz*id>=n?n:bsz*id;
}
priority_queue<int>num[maxb+5];//大根堆,存储每一块中的数
priority_queue<int,vector<int>,greater<int> >mark[maxb+5] ;//小根堆,存储每一个块上替换标记
//从小到大是因为序列中的数会被优先换成小的标记
void rebuild(int id){//重构每个块,不管是整块替换还是部分替换,都要调用rebuild
while(!num[id].empty()) num[id].pop();
for(int i=lb(id);i<=rb(id);i++){
num[id].push(a[i]);
}
}
void push_down(int id){
if(!mark[id].empty()){
for(int i=lb(id);i<=rb(id);i++){
int v=mark[id].top();
if(v<a[i]){
swap(a[i],v);
mark[id].pop();
mark[id].push(v);//把a[i]与v交换
}
}
while(!mark[id].empty()) mark[id].pop();
rebuild(id);
} } int update(int l,int r,int A){
push_down(bel[l]);//非整块操作,必须下推标记
for(int i=l;i<=min(r,rb(bel[l]));i++){
if(A<a[i]) swap(a[i],A);
}
rebuild(bel[l]);
for(int i=bel[l]+1;i<bel[r];i++){
int v=num[i].top();
if(v>A){ //如果最大值比A小,就不替换了
num[i].pop();
num[i].push(A);
mark[i].push(A);
swap(A,v);
//整块修改的时候只需要替换最大值,剩下的操作push_down的时候完成
}
}
if(bel[l]!=bel[r]){
push_down(bel[r]);
for(int i=lb(bel[r]);i<=r;i++){
if(A<a[i]) swap(a[i],A);
}
rebuild(bel[r]);
}
return A;
}
int main(){
int l,r,A;
qread(n);
qread(m);
for(int i=1;i<=n;i++) qread(a[i]);
bsz=sqrt(n);
for(int i=1;i<=n;i++){
bel[i]=(i-1)/bsz+1;
num[bel[i]].push(a[i]);
}
for(int i=1;i<=m;i++){
qread(l);
qread(r);
qread(A);
if(l<=r){
A=update(l,r,A);
}else{
A=update(l,n,A);
A=update(1,r,A);
}
// printf("db:");
qprint(A);
putchar('\n');
}
}

[LOJ2736] [JOISC 2016 Day 3] 回转寿司 (分块+堆)的更多相关文章

  1. BZOJ_4627_[BeiJing2016]回转寿司_离散化+树状数组

    BZOJ_4627_[BeiJing2016]回转寿司_离散化+树状数组 Description 酷爱日料的小Z经常光顾学校东门外的回转寿司店.在这里,一盘盘寿司通过传送带依次呈现在小Z眼前.不同的寿 ...

  2. Loj #2731 「JOISC 2016 Day 1」棋盘游戏

    Loj 2731 「JOISC 2016 Day 1」棋盘游戏 JOI 君有一个棋盘,棋盘上有 \(N\) 行 \(3\) 列 的格子.JOI 君有若干棋子,并想用它们来玩一个游戏.初始状态棋盘上至少 ...

  3. bzoj 4627: [BeiJing2016]回转寿司 -- 权值线段树

    4627: [BeiJing2016]回转寿司 Time Limit: 10 Sec  Memory Limit: 256 MB Description 酷爱日料的小Z经常光顾学校东门外的回转寿司店. ...

  4. bzoj 4627: [BeiJing2016]回转寿司

    4627: [BeiJing2016]回转寿司 Description 酷爱日料的小Z经常光顾学校东门外的回转寿司店.在这里,一盘盘寿司通过传送带依次呈现在小Z眼前.不同的寿 司带给小Z的味觉感受是不 ...

  5. 【BZOJ4627】[BeiJing2016]回转寿司 SBT

    [BZOJ4627][BeiJing2016]回转寿司 Description 酷爱日料的小Z经常光顾学校东门外的回转寿司店.在这里,一盘盘寿司通过传送带依次呈现在小Z眼前.不同的寿司带给小Z的味觉感 ...

  6. K - 回转寿司(值域段数(板题) + 动态开点)

    回转寿司 Description 酷爱日料的小Z经常光顾学校东门外的回转寿司店.在这里,一盘盘寿司通过传送带依次呈现在小Z眼前.不同的寿 司带给小Z的味觉感受是不一样的,我们定义小Z对每盘寿司都有一个 ...

  7. 「JOISC 2016 Day 1」棋盘游戏

    「JOISC 2016 Day 1」棋盘游戏 先判无解:第1,3行有连续的空格或四个角有空格. 然后可以发现有解的情况第1,3行可以在任意时间摆放. 对于某一列,若第2行放有棋子,那么显然可以把棋盘分 ...

  8. LOJ 2736 「JOISC 2016 Day 3」回转寿司 ——堆+分块思路

    题目:https://loj.ac/problem/2736 如果每个询问都是 l = 1 , r = n ,那么每次输出序列的 n 个数与本次操作的数的最大值即可.可以用堆维护. 不同区间的询问,可 ...

  9. 「JOISC 2016 Day 3」回转寿司

    https://loj.ac/problem/2736 题解 挺有意思的题. 考虑这种操作不好直接维护,还有时限比较长,所以考虑分块. 考虑一个操作对整个块的影响,无非就是可能把最大的拿走,再把新的元 ...

随机推荐

  1. java.sql.SQLException: Unknown system variable 'query_cache_size'

    改为 <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java< ...

  2. 【Linux】CentOS6上安装Python3.7(config、make、make install)及“No module named '_ctypes'”/pip install时“ssl module in Python is not available.”的解决

    1.下载安装包 https://www.python.org/ftp/python/ 该目录下选择所需要的版本进行下载.解压. wget https://www.python.org/ftp/pyth ...

  3. vue全家桶是啥?

    Vue有著名的全家桶系列,包含了 1,调试插件:可以选择 Chrome 插件 vue Devtool(需要下载工具包).打开控制台选择 vue 面板.也可以选择 Vuex 选项.vuex(http:/ ...

  4. Bugku 杂项 又一张图片,还单纯吗

    又一张图片,还单纯吗 下载后,用binwalk打开图片 使用foremost 2.png进行分离 得到图片 关于foremost foremost [-v|-V|-h|-T|-Q|-q|-a|-w-d ...

  5. 3D Computer Grapihcs Using OpenGL - 01 环境设置

    这系列文章是我学习Youtube上一套OpenGL教程的笔记,自己对教程的案例重新制作并且做了一定程度的修改(更有条理,且修正了一些问题).后续将持续更新. Visual Studio 2017工程 ...

  6. React 进阶设计与控制权问题

    控制权--这个概念在编程中至关重要.比如,"轮子"封装层与业务消费层对于控制权的"争夺",就是一个很有意思的话题.这在 React 世界里也不例外.表面上看,我 ...

  7. vue双向绑定原理(简单实现原理附demo)

    先上效果图 简单的实现数据的双向绑定首先来了解一个东西:Object.defineProperty() https://developer.mozilla.org/zh-CN/docs/Web/Jav ...

  8. python3下multiprocessing、threading和gevent性能对比----暨进程池、线程池和协程池性能对比

    python3下multiprocessing.threading和gevent性能对比----暨进程池.线程池和协程池性能对比   标签: python3 / 线程池 / multiprocessi ...

  9. GNU Makefile手札

    通配符 $@ 目标集合 $< 第一个依赖文件 $^ 所有依赖文件 $? 执行结果 % 表示任意长度的字符 $$$$ 随机数 @<command> 执行时不输出该命令到控制台 变量赋值 ...

  10. (转)flexpaper 参数

    本文转载自:http://blog.csdn.net/z69183787/article/details/18659913 Flexpaper可能用到如下参数   SwfFile (String) 需 ...