[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. 获取sender进程所产生的trace文件

    直接开启数据库实例级别的sql_trace是不现实的,会对所有连接到oracle的session都产生sql_trace,这样会产生大量的trace文件(垃圾文件),因此为跟踪特定进程的sql信息,可 ...

  2. 原生JS 将时间转换成几秒前,几分钟前…常用于评论回复功能

    //格式化时间 留备用~ function getDateDiff(dateStr) { var publishTime = dateStr / 1000, d_seconds, d_minutes, ...

  3. MongoDB的$作为下标的用法

    在MongoDB中有一个非常神奇的符号 "$" "$"  在 update 中 加上关键字 就 变成了 修改器 其实 "$" 字符 独立出现 ...

  4. R语言 eval(quote(x)) 和 eval(x)

    eval() 's first argument is an expression. So if you only provide one argument, it will evaluate the ...

  5. SpringCloud学习系列-构建部门微服务消费者Module

    1.新建microservicecloud-consumer-dept-80 2.Pom <project xmlns="http://maven.apache.org/POM/4.0 ...

  6. LeetCode--072--编辑距离(python)

    给定两个单词 word1 和 word2,计算出将 word1 转换成 word2 所使用的最少操作数 . 你可以对一个单词进行如下三种操作: 插入一个字符删除一个字符替换一个字符示例 1: 输入: ...

  7. react native之封装离线缓存框架

    请求数据=>本地有无缓存+缓存数据是否过期 =>可用 =>不可用 将代码封装成一个DataStore.js文件, 这里面主要提供:从本地获取数据,从网络获取数据,创建本地时间戳,请求 ...

  8. React Native 之react-native-sqlite-storage

    npm 官网指导: https://www.npmjs.com/package/react-native-sqlite-storage 1. 执行: npm install react-native- ...

  9. HDU-6709 Fishing Master

    Description Heard that eom is a fishing MASTER, you want to acknowledge him as your mentor. As every ...

  10. React-Native 之 GD (十八)监听 TabBarItem 点击与传值实现 点击 Item 进行刷新功能

    监听 TabBarItem 点击与传值实现 点击 Item 进行刷新功能 原版 APP 中当我们点击 首页和海淘 2个 Item 时,会马上获取最新数据个数然后进行更新,这边来实现一下这个功能. 1. ...