P7706 「Wdsr-2.7」文文的摄影布置
题意
给定长度为 \(n\) 的数组 \(a\) 和 \(b\),支持单点修改,\(q\) 次区间查询 \(\max_{l\le i<k\le r} \{a_i + a_k - \min_{i<j<k}b_j\}\)。
\(n,q\le 5\times10^5,1\le a_i,b_i\le10^8\qquad\text{2s,256MB}\)
题解
考虑使用线段树维护信息。记 \(\psi(p)\) 表示线段树上节点 \(p\) 的答案,接下来要考虑如何根据左右子树的答案,更新当前节点的答案。
注意到答案和 \(i,j,k\) 三个位置有关,我们希望 \(a_i,a_k\) 尽量的大,\(b_j\) 尽量小,那么就需要维护区间 \(a\) 最大值,\(b\) 最小值。
然后是分类讨论,我们讨论 \(\psi(p)\) 最大时,它的 \(i,j,k\) 会落在什么位置:
- \(i,j,k\) 均在左子树或均在右子树:直接取左右子树答案最大值。\(\psi(p) = \max(\psi(ls),\psi(rs))\)。
- \(i,j\) 在左子树,\(k\) 在右子树:\(k\) 直接选右子树中 \(b_k\) 最大的,对于 \(i,j\),需要的是 \(\max_{i<j}a_i - b_j\)。
- \(i\) 在左子树,\(j,k\) 在右子树:同上,\(i\) 选最大的,\(j,k\) 是 \(\max_{j<k}- b_j + a_k\)。
发现还需要维护 \(\max_{i<j}a_i - b_j\) 这样的信息,继续分类讨论:
- \(i,j\) 在同一子树中:取左右子树最大值。
- \(i\) 在左子树,\(j\) 在右子树:\(i\) 取左子树 \(a_i\) 最大值,\(j\) 取右子树 \(b_j\) 最小值。
对于 \(\max_{j < k}-b_j + a_k\),同样的讨论。
记第一种信息为 \(L(p)\),第二种信息为 \(R(p)\),\(a\) 最大为 \(Max(p)\),\(b\) 最小为 \(Min(p)\),我们就能整合出以下的式子:
\]
\]
\]
一环套一环,太牛了哥!
单点修改就是直接对 \(Min,Max\) 进行修改,更新信息;查询就把所有区间像上面的方式合并起来,复杂度是一只 \(\log\) 的。
时间复杂度 \(\mathcal{O}(n\log n)\),分块 \(\mathcal{O}(n\sqrt{n})\) 卡卡应该能过。
代码实现中,我们可以采取重载 \(+\) 运算进行信息的维护,这样就可以模仿区间加的方式维护信息:
XDT operator+ (XDT A){
XDT res;
res.l = min(l,A.l);
res.r = max(r,A.r);
res.Max = max(Max,A.Max);
res.Min = min(Min,A.Min);
res.L = max(max(L,A.L),Max - A.Min);
res.R = max(max(R,A.R),- Min + A.Max);
res.Ans = max(max(Ans,A.Ans),max(L + A.Max,Max + A.R));
return res;
}
完整代码如下:
#include <bits/stdc++.h>
using namespace std;
const int N = 5e5 + 5;
int n,q,a[N],b[N];
struct XDT{
int l,r;
int Ans,Max,Min,L,R;
XDT() { Ans = Max = Min = L = R = - 1e9; }
XDT operator+ (XDT A){
XDT res;
res.l = min(l,A.l);
res.r = max(r,A.r);
res.Max = max(Max,A.Max);
res.Min = min(Min,A.Min);
res.L = max(max(L,A.L),Max - A.Min);
res.R = max(max(R,A.R),- Min + A.Max);
res.Ans = max(max(Ans,A.Ans),max(L + A.Max,Max + A.R));
return res;
}
}t[N << 2];
#define ls p << 1
#define rs p << 1 | 1
void pushup(int p){
t[p] = t[ls] + t[rs];
}
void build(int p,int l,int r){
if (l == r){
t[p].l = t[p].r = l;
t[p].Max = a[l];
t[p].Min = b[l];
return ;
}
int mid = (l + r) >> 1;
build(ls,l,mid), build(rs,mid + 1,r);
pushup(p);
}
void modify1(int p,int x,int y){
if (t[p].l == t[p].r){
t[p].Max = y;
return ;
}
int mid = (t[p].l + t[p].r) >> 1;
if (x <= mid) modify1(ls,x,y);
else modify1(rs,x,y);
pushup(p);
}
void modify2(int p,int x,int y){
if (t[p].l == t[p].r){
t[p].Min = y;
return ;
}
int mid = (t[p].l + t[p].r) >> 1;
if (x <= mid) modify2(ls,x,y);
else modify2(rs,x,y);
pushup(p);
}
void query(int p,int l,int r,XDT &ans){
if (l <= t[p].l && t[p].r <= r)
return (void)(ans = ans + t[p]);
int mid = (t[p].l + t[p].r) >> 1;
if (l <= mid) query(ls,l,r,ans);
if (r > mid) query(rs,l,r,ans);
}
int read(){
int x = 0; char ch = getchar();
while (ch < '0' || ch > '9'){
ch = getchar();
}
while ('0' <= ch && ch <= '9'){
x = (x << 1) + (x << 3) + (ch ^ 48);
ch = getchar();
}
return x;
}
int main(){
n = read(), q = read();
for (int i = 1;i <= n;i++) a[i] = read();
for (int i = 1;i <= n;i++) b[i] = read();
build(1,1,n);
while (q--){
int o = read(), l = read(), r = read();
XDT ans;
if (o == 1) modify1(1,l,r);
if (o == 2) modify2(1,l,r);
if (o == 3) query(1,l,r,ans), printf("%d\n",ans.Ans);
}
return 0;
}
P7706 「Wdsr-2.7」文文的摄影布置的更多相关文章
- 「查缺补漏」巩固你的Nginx知识体系
Nginx篇 基本介绍 Nginx是一款轻量级的 Web服务器 / 反向代理服务器 / 电子邮件(IMAP/POP3)代理服务器,主要的优点是: 支持高并发连接,尤其是静态界面,官方测试Nginx能够 ...
- 「编程羽录」上线,程序员必备的这些技能你能get到嘛?
大家好,我是小羽. 好久不见,给大家带来个好消息,小羽的全新专题「编程羽录」系列正式上新,主要是介绍一些关于面试题和经验总结的文章. 会为大家提供一些技术栈之外,程序员还需要的其他方面硬核知识,做到全 ...
- 对于前端,「微信小程序」其实不美好
微信小程序开放公测了,9月底我曾经写过一篇 「微信小程序」来了,其中最后一句:"谢天谢地,我居然还是个前端". 这种火爆的新事物总是令人激动,感谢这个时代. 但是,当我真作为开发者 ...
- macOS安装「oh my zsh」
目前常用的 Linux 系统和 OS X 系统的默认 Shell 都是 bash,但是真正强大的 Shell 是深藏不露的 zsh, 这货绝对是马车中的跑车,跑车中的飞行车,史称『终极 Shell』, ...
- 报名|「OneAPM x DaoCloud」技术公开课:Docker性能监控!
如今,越来越多的公司开始 Docker 了,「三分之二的公司在尝试了 Docker 后最终使用了它」,也就是说 Docker 的转化率达到了 67%,同时转化时长也控制在 60 天内. 既然 Dock ...
- 企业运营对 DevOps 的「傲慢与偏见」
摘要:出于各种原因,并非所有人都信任 DevOps .有些人觉得 DevOps 只不过给开发者改善产品提供了一个途径而已,还有的人觉得 DevOps 是一堆悦耳的空头支票,甚至有人认为 DevOps ...
- 「前端开发者」如何把握住「微信小程序」这波红利?
由于前两周一直在老家处理重要事情,虽然朋友圈被「微信小程序」刷爆了,但并没有时间深入了解. 昨天回广州之后,第一件事情就是把「微信小程序」相关的文章.开发文档.设计规范全部看了一遍,基本上明白了「微信 ...
- 「花田对」CSDN程序员专场——谁来拯救技术宅!_豆瓣
「花田对」CSDN程序员专场--谁来拯救技术宅!_豆瓣 「花田对」CSDN程序员专场--谁来拯救技术宅!
- Objective-C 实用关键字详解1「面试、工作」看我就 🐒 了 ^_^.
在写项目 或 阅读别人的代码(一些优秀的源码)中,总能发现一些常见的关键字,随着编程经验的积累大部分还是知道是什么意思 的. 相信很多开发者跟我当初一样,只是基本的常用关键字定义属性会使用,但在关键字 ...
- LOJ6003 - 「网络流 24 题」魔术球
原题链接 Description 假设有根柱子,现要按下述规则在这根柱子中依次放入编号为的球. 每次只能在某根柱子的最上面放球. 在同一根柱子中,任何2个相邻球的编号之和为完全平方数. 试设计一个算法 ...
随机推荐
- 树莓派4B-高精度驱动步进电机
树莓派4B-高精度驱动步进电机 项目介绍 利用4B树莓派控制步进电机转动,精度可达:0.0144度 (即360度/25000) 适用于非常精密的角度转动. 舵机的精度为1度,无法实现超高精度控制. 硬 ...
- Java 面向对象编程之接口
什么是接口? 是抽象方法的集合,接口通常以interface来声明,一个类通过继承接口的方式,从而来继承接口的抽象方法 语法 interface 名称 [extends 其他的接⼝名] { // 声明 ...
- Python爬虫(1-4)-基本概念、六个读取方法、下载(源代码、图片、视频 )、user-agent反爬
Python爬虫 一.爬虫相关概念介绍 1.什么是互联网爬虫 如果我们把互联网比作一张大的蜘蛛网,那一台计算机上的数据便是蜘蛛网上的一个猎物,而爬虫程序就是一只小蜘蛛,沿着蜘蛛网抓取自己想要的数据 解 ...
- Python 正则表达式实战之Java日志解析
需求描述 基于生产监控告警需求,需要对Java日志进行解析,提取相关信息,作为告警通知消息的内容部分. 提取思路 具体怎么提取,提取哪些内容呢?这里笔者分析了大量不同形态的生产日志,最后总结出4种形态 ...
- 利用Elasticsearch实现地理位置、城市搜索服务
最近用到一些简单的地理位置查询接口,基于当前定位获取用户所在位置信息(省市区),然后基于该信息查询当前区域的......提供服务. 然后就自己研究了下GIS,作为一个程序员.自己能不能实现这个功能呢? ...
- scratch源码下载 | 蜘蛛传说
程序说明: <蜘蛛传说>是一个通过Scratch平台制作的互动游戏项目.在这个故事中,玩家将扮演一只蜘蛛,其原本和平的生活被一只入侵的壁虎所打破.为了保卫自己的家园,蜘蛛必须运用智慧和勇气 ...
- 查询当前网段的所有在用IP
查询当前网段的所有在用IP For /L %i in (0,1,254) DO ping 192.168.10.%i >>D:\IP.txt https://www.cnblogs.com ...
- 【Vue】Re22 Axios
Axios[AJAX I\O System] 创建案例项目并且安装Axios npm install axios --save 接口测试网址: http://httpbin.org/ 案例提供的数据地 ...
- 【SpringBoot】Re 01 补充学习
对SpringBoot后续的再补充学习: 使用IDEA创建项目不勾选任何组件 默认的POM结构: <?xml version="1.0" encoding="UTF ...
- 如何为华为超算平台设置cuda路径
在提交主机上修改.bashrc文件: 第一种: 使用运行主机上的cuda环境: # CUDAexport PATH=/usr/local/cuda-11.4/bin:$PATHexport LD_LI ...