TAMAYA

挺有意思的维护题。

题面

n个小夫坐成一排,每个小夫有一个真实值vi。小夫们有m场聚会,第i次聚会会在编号为 [li, ri] 的小夫中举办。

聚会之后,这些小夫的真实值会变为他们之中的真实值的最大值。将会发生q次事件,有两类事件。

  • 第一类事件,第x个小夫的真实值变成了y。
  • 第二类事件,小夫们向你提问,假如第L次聚会到第R次聚会按顺序举办,第x个 小夫的真实值会变成多少。

对于所有测试点:\(n,m,q\leq 5\times 10^5,l_i\leq r_i\leq n,x\leq n,L\leq R,op\in\{1,2\},v_i,y\leq 10^9\)

思路

下文称聚会为操作,提问为查询。

只考虑 \(op=2\) 的情况,对于一个询问中的 \(x_i\),令 \([lx,rx]=[x_i,x_i]\)。

倒序进行询问中的操作 \([L_i,R_i]\),若一次操作 \([l_j,r_j](j\in [L_i,R_i])\) 与现在的 \([lx,rx]\) 有交集,那么令 \(lx=\min(lx,l_j)\),\(rx=\max(rx,r_j)\);形式化的,若 \([l_j,r_j]\cap[lx,rx]\neq \emptyset\),那么令 \([lx,rx]=[l_j,r_j]\cup[lx,rx]\)。

最后的答案就是 \([lx,rx]\) 区间中的最大值。

考虑到每一个查询的 \([x_i,x_i]\),在第一次取并集后,会以一个操作中的区间 \([l_j,r_j]\) 的形式出现。考虑对于一个操作 \(j\),维护出执行了倒叙执行操作 \(k\sim j(k<j)\) 区间左端点和右端点的变化。

但是这样的复杂度来到了 \(O(m^2)\)。

考虑倍增维护,维护出合并 \(2^k\) 次操作后的区间的左右端点的变化。

由于 \(l_j\) 合并只会变成前面某一次操作的 \(l\),这样就可以倍增维护出变成了哪一次操作的 \(l\)。

\(r\) 同理。

最后对于查询 \(x_i\),维护出最后一次被操作覆盖即可。

CODE

#include<bits/stdc++.h>
using namespace std; const int maxn=5e5+5; int n,m,q;
int v[maxn],pl[maxn][25],pr[maxn][25],l[maxn],r[maxn],L[maxn],R[maxn],id[maxn],op[maxn],x[maxn]; vector<int>vec[maxn]; namespace linetree
{
#define lch(p) p*2
#define rch(p) p*2+1
struct treenode{int mx,lazy;}tr[maxn*20];
inline void updata(int p) {tr[p].mx=max(tr[lch(p)].mx,tr[rch(p)].mx);}
inline void push_down(int p,int l,int r)
{
if(!tr[p].lazy) return ;
if(l==r) return ;
tr[lch(p)].mx=max(tr[p].lazy,tr[lch(p)].mx);
tr[lch(p)].lazy=max(tr[p].lazy,tr[lch(p)].lazy);
tr[rch(p)].mx=max(tr[p].lazy,tr[rch(p)].mx);
tr[rch(p)].lazy=max(tr[p].lazy,tr[rch(p)].lazy);
tr[p].lazy=0;
}
inline void change(int p,int l,int r,int lx,int rx,int v)
{
if(r<lx||l>rx) return ;
push_down(p,l,r);
if(lx<=l&&r<=rx)
{
tr[p].mx=v;tr[p].lazy=v;
return ;
}
int mid=(l+r)>>1;
change(lch(p),l,mid,lx,rx,v);change(rch(p),mid+1,r,lx,rx,v);
updata(p);
}
inline int qry(int p,int l,int r,int lx,int rx)
{
if(r<lx||l>rx) return 0;
push_down(p,l,r);
if(lx<=l&&r<=rx) return tr[p].mx;
int mid=(l+r)>>1;
return max(qry(lch(p),l,mid,lx,rx),qry(rch(p),mid+1,r,lx,rx));
}
inline void clr(){memset(tr,0,sizeof(tr));}
} int main()
{
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++) scanf("%d",&v[i]);
for(int i=1;i<=m;i++) scanf("%d%d",&l[i],&r[i]);
scanf("%d",&q);
for(int i=1;i<=q;i++)
{
scanf("%d%d%d",&op[i],&L[i],&R[i]);
if(op[i]==2) scanf("%d",&x[i]),vec[R[i]].emplace_back(i);
}
for(int i=1;i<=m;i++)
{
pl[i][0]=linetree::qry(1,1,n,l[i],l[i]);
pr[i][0]=linetree::qry(1,1,n,r[i],r[i]);
linetree::change(1,1,n,l[i],r[i],i);
for(auto v:vec[i]) id[v]=linetree::qry(1,1,n,x[v],x[v]);
}
for(int i=1;i<=m;i++) for(int j=1;j<=log2(m);j++)
{
pl[i][j]=pl[pl[i][j-1]][j-1];
pr[i][j]=pr[pr[i][j-1]][j-1];
}
linetree::clr();
for(int i=1;i<=n;i++) linetree::change(1,1,n,i,i,v[i]);
for(int i=1;i<=q;i++)
{
if(op[i]==1)
linetree::change(1,1,n,L[i],L[i],R[i]);
else
{
if(id[i]<L[i]) {printf("%d\n",linetree::qry(1,1,n,x[i],x[i]));continue;}
int lx=id[i],rx=id[i];
for(int j=log2(m);~j;j--) if(pl[lx][j]>=L[i]) lx=pl[lx][j];
for(int j=log2(m);~j;j--) if(pr[rx][j]>=L[i]) rx=pr[rx][j];
printf("%d\n",linetree::qry(1,1,n,l[lx],r[rx]));
}
}
}

随机推荐

  1. String, StringBuffer 和 StringBuilder之间的区别

    String, StringBuffer 和 StringBuilder 可变性 String不可变 StringBuffer 和 StringBuilder 可变 线程安全 String 不可变,因 ...

  2. 使用 SSH 转义代码来控制连接

    OpenSSH 最常被忽视的一个非常有用的功能是能够从连接内部控制会话的某些方面.通过使用 SSH 转义代码,我们能够在会话内部与本地 SSH 软件进行交互. 强制从客户端断开连接(如何退出卡住或冻结 ...

  3. 将workbench 导出的sql数据修改为 oracle 的sql版本

    将导出的文件内容复制到 dd1.txt,或其它文件,修改path的值即可 修改后的sql文件为 dd1.sql : 替换的内容: 1. 全局替换了一些字符串,如` 2. workbench导出的sql ...

  4. 【YashanDB知识库】含有NUL字节的varchar字符串查询时出现截断

    [问题分类]功能使用 [关键字]NUL字符 [问题描述]数据库中插入一条含有NUL字符(\00)的字符串,使用yasql在终端进行查询,字符串从NUL处被截断,未能完整展示全部字符. [问题原因分析] ...

  5. 示例python 批量操作excel统计销售榜品牌及销售额

    示例统计销售榜品牌及销售额 import pandas as pd import numpy as np import os os.chdir('F:\\50mat\源数据1000张表格') name ...

  6. ComfyUI 基础教程(二) —— Stable Diffusion 文生图基础工作流及常用节点介绍

    上一篇文章讲解述首次启动 ComfyUI 会自动打开一个最基础的文生图工作流.实际上,后续我们可以通过菜单选项,或者快捷键 ctrl + D来打开这个默认工作流.默认工作流如下: 这是一个最基础的文生 ...

  7. AI工具合集

    以下工具来源于互联网,可能会失效,请参考使用 网红工具 名称 链接 说明   GPT-4 https://chat.openai.com/ 需要梯子,需要付费.功能最强大的聊天机 器人. 文心一言 h ...

  8. Mininet安装记录

    安装环境: Ubuntu虚拟机版本:14.04 Mininet版本:2.3.1b1 1.更改软件镜像源 在设置中进行如下操作: 选择国内的镜像站点,如阿里云. 点击关闭后,在弹出的窗口中点击重新载入, ...

  9. C#上位机与PLC通信心跳的实现方法

    -Begin- 大家好!我是付工.众所周知,在工业自动化控制系统中,上位机与下位机之间的通信是实现自动化生产的关键环节之一.为了确保通信的稳定性和可靠性,我们通用会采用一种被称为[心跳机制]的方法,它 ...

  10. 解决 Vue 项目打包上线后客户端缓存的问题

    由于重新打包后会导致对应的 js 和 css 文件 hash 值发生变化,客户端不刷新的话就会存在之前的文件找不到,导致报错的问题. 通过 build.sh 定义打包命令 #!/usr/bin/env ...