@bzoj - 2388@ 旅行规划
@description@
请你维护一个序列,支持两种操作:
(1)某个区间 [x, y] 内的数同时加上一个增量 k。
(2)询问某一个区间 [x, y] 中从 1 开始的最大前缀和。
input
第一行给出一个整数 n。n <= 100000。接下来一行 n 个整数表示序列的初始值。
第三行给出一个整数 m,m <= 100000。接下来 m 行每行一个操作。
(1) 0 x y k:表示给区间 [x, y] 同时加上 k。
(2) 1 x y:询问区间 [x, y]。
output
对于每个询问,输出一个整数表示最大前缀和。
sample input
5
1 8 -8 3 -7
3
1 1 5
0 1 3 6
1 2 4
sample output
9
22
@solution@
我们考虑直接维护前缀和序列,则操作(2)就是在查询区间最大值。
而对于操作(1),我们相当于两部分操作:
对于 x <= i <= y,给 i 位置加上 (i-x+1)*k;对于 y < i,给 i 位置加上 (y-x+1)*k。
前一个可以拆成 (-x+1)*k + i*k,是常数 + 系数*位置的形式;后面那个也可以看成这种形式,只是位置前面的系数为 0。
看起来还是不好维护,但是我们可以注意到这样一件事情:对于某一个位置 i,它的值总是形如 k*i + b 的形式。
直线解析式。所以我们考虑用几何方法来维护这种东西。几何方法当然首先想到凸包。
每次操作相当于给区间内所有点的斜率与截距同时加上增量,手算一下会发现这个区间相邻两点之间的斜率也会同时增加 k,这样也就是说这个区间的凸包形状不会变化。
线段树不太好搞(况且这道题时限 50s ),我们考虑使用分块算法来维护凸包。
修改时,散块暴力修改并重构凸包,整块打标记,记录这个区间整体的斜率与截距变化量。
查询时,散块暴力求答案,整块凸包上二分。
时间复杂度 \(O(n\sqrt{n}\log n)\)
@accepted code@
#include<cstdio>
#include<algorithm>
using namespace std;
typedef long long ll;
const int MAXN = 100000;
const int BLOCK = 320;
const ll INF = (1ll<<62);
ll sp[BLOCK + 5], b1[MAXN + 5], b2[BLOCK + 5];
int le[BLOCK + 5], ri[BLOCK + 5], num[MAXN + 5];
int stk[MAXN + 5], tp[BLOCK + 5], n, m, bcnt = 0;
ll get_ans(int x) {
return sp[num[x]]*x + b1[x] + b2[num[x]];
}
ll query(int x) {
int l = le[x], r = tp[x];
while( l < r ) {
int mid = (l + r) >> 1;
if( get_ans(stk[mid]) >= get_ans(stk[mid+1]) ) r = mid;
else l = mid + 1;
}
return get_ans(stk[r]);
}
void push_tag(int x) {
for(int i=le[x];i<=ri[x];i++)
b1[i] = get_ans(i);
sp[x] = b2[x] = 0;
}
void build(int x) {
tp[x] = le[x] - 1;
for(int i=le[x];i<=ri[x];i++) {
while( tp[x] > le[x] && (get_ans(i) - get_ans(stk[tp[x]]))*(stk[tp[x]] - stk[tp[x] - 1]) >= (get_ans(stk[tp[x]])-get_ans(stk[tp[x] - 1]))*(i - stk[tp[x]]) )
tp[x]--;
stk[++tp[x]] = i;
}
}
void init() {
for(int i=0;i<n;i++) {
if( i % BLOCK == 0 ) {
num[i] = (++bcnt);
le[num[i]] = ri[num[i]] = i;
sp[num[i]] = b2[num[i]] = 0;
}
else ri[num[i] = bcnt]++;
}
}
int main() {
scanf("%d", &n); init();
for(int i=0;i<n;i++)
scanf("%lld", &b1[i]), b1[i] += b1[i-1];
for(int i=1;i<=bcnt;i++)
build(i);
scanf("%d", &m);
for(int i=1;i<=m;i++) {
int op, x, y; ll k;
scanf("%d%d%d", &op, &x, &y);
x--, y--;
if( op == 0 ) {
scanf("%lld", &k);
if( num[x] != num[y] ) {
push_tag(num[x]), push_tag(num[y]);
for(int i=x;i<=ri[num[x]];i++)
b1[i] += k*(i-x+1);
for(int i=le[num[y]];i<=y;i++)
b1[i] += k*(i-x+1);
for(int i=y+1;i<=ri[num[y]];i++)
b1[i] += k*(y-x+1);
build(num[x]), build(num[y]);
for(int i=num[x]+1;i<=num[y]-1;i++)
sp[i] += k, b2[i] += k*(-x+1);
}
else {
push_tag(num[x]);
for(int i=x;i<=y;i++)
b1[i] += k*(i-x+1);
for(int i=y+1;i<=ri[num[y]];i++)
b1[i] += k*(y-x+1);
build(num[x]);
}
for(int i=num[y]+1;i<=bcnt;i++)
b2[i] += k*(y-x+1);
}
else {
ll ans = -INF;
if( num[x] != num[y] ) {
for(int i=x;i<=ri[num[x]];i++)
ans = max(ans, get_ans(i));
for(int i=le[num[y]];i<=y;i++)
ans = max(ans, get_ans(i));
for(int i=num[x]+1;i<=num[y]-1;i++)
ans = max(ans, query(i));
}
else {
for(int i=x;i<=y;i++)
ans = max(ans, get_ans(i));
}
printf("%lld\n", ans);
}
}
}
@details@
突然发现这是我第一次写分块?原来我以前从来没写过这种东西?
把分块的左端点右端点以及每个点属于哪个块先预处理出来感觉比较好写。
并且把块的大小设置为常数也是一个不错的懒人做法(虽然想想都知道这样肯定常数大)。
@bzoj - 2388@ 旅行规划的更多相关文章
- BZOJ 2388: 旅行规划 [分块 凸包 等差数列]
传送门 题意: 区间加和询问一段区间内整体前缀和的最大值 刚才还在想做完这道题做一道区间加等差数列结果发现这道就是.... 唯一的不同在于前缀和一段区间加上等差数列后,区间后面也要加上一个常数!!! ...
- BZOJ 2388--旅行规划(分块&单调栈&二分)
2388: 旅行规划 Time Limit: 50 Sec Memory Limit: 128 MBSubmit: 405 Solved: 118[Submit][Status][Discuss] ...
- bzoj 4501: 旅行 01分数规划+概率期望dp
题目大意: http://www.lydsy.com/JudgeOnline/problem.php?id=4501 题解: 首先我们不考虑可以删除边的情况下,如何计算期望边数. 然后我们发现这是个有 ...
- bzoj 3531 旅行
动态开点线段树+树链剖分 对于每一种宗教信仰都开一颗线段树 空间: QlogN 即每一次修改都只会改变logN 个点 时间 O(QlogN) naive题 边没有开两倍 QAQ bzoj 35 ...
- bzoj 4501 旅行
01分数规划+最大权闭合子图 倒拓扑序处理每个节点 $$f[x]=\frac{\sum{f[v]}}{n}+1$$ 二分答案$val$ 只需要判断是否存在$\sum{f[v]}+1-val>0$ ...
- 旅行规划(travel)
题目描述 OIVillage 是一个风景秀美的乡村,为了更好的利用当地的旅游资源,吸引游客,推动经济发展,xkszltl 决定修建了一条铁路将当地 nnn 个最著名的经典连接起来,让游客可以通过火车从 ...
- BZOJ2388:旅行规划(travel)——分块凸包
题目 OIVillage 是一个风景秀美的乡村,为了更好的利用当地的旅游资源,吸引游客,推动经济发展,xkszltl 决定修建了一条铁路将当地 $n$ 个最著名的经典连接起来,让游客可以通过火车从铁路 ...
- BZOJ 4464 旅行时的困惑 最小流
题面: Waldives 有 N 个小岛.目前的交通系统中包含 N-1 条快艇专线,每条快艇 专线连接两个岛.这 N-1条快艇专线恰好形成了一棵树. 由于特殊的原因,所有N-1条快艇专线都是单向的.这 ...
- 「BZOJ2388」旅行规划
传送门 分块+凸包 求出前缀和数组s 对于l~r加上k,相当于s[l]~s[r]加上一个首项为k,公差为k的等差数列.r~n加上k*(r-l+1). 分块之后对每一块维护两个标记,一个记录它加的等差数 ...
随机推荐
- scanf("%c", &ch)和scanf(" %c", &ch)和scanf("%s", str)的注意事项
scanf("%c", &ch)和scanf(" %c", &ch): %c会读取回车和空格,所以一定要使用后者,即在%c前面加一个空格. %s ...
- 2019-2-14-VisualStudio-通过外部调试方法快速调试库代码
title author date CreateTime categories VisualStudio 通过外部调试方法快速调试库代码 lindexi 2019-2-14 22:1:37 +0800 ...
- Vbulletin Used to Show Malicious Advertisements
In the past, we have seen a massive amount of vBulletin websites compromised through theVBSeo Vulner ...
- 避免闲置云资源浪费 | 阿里云轻量级分布式应用服务 SAE 邀您公测
您是否遇到过: 资源利用率低,多数服务器CPU平均利用率在10%以下,用户需为大量闲置资源买单. 感知 IaaS 购买和集群运维,人员技能要求高,运维效率低. 想拥抱 Kubernetes.微服务架构 ...
- Faster RCNN算法训练代码解析(1)
这周看完faster-rcnn后,应该对其源码进行一个解析,以便后面的使用. 那首先直接先主函数出发py-faster-rcnn/tools/train_faster_rcnn_alt_opt.py ...
- select @@identity的用法 转
用select @@identity得到上一次插入记录时自动产生的ID 如果你使用存储过程的话,将非常简单,代码如下:SET @NewID=@@IDENTITY 说明: 在一条 INSERT.SELE ...
- hack 记录
0.寻找信号强的wifi,对于隐藏ssid 的可通过嗅探 1.wifi密码:wifi万能钥匙.minidwep-gtk.aircrack-ng 2.对于绑定mac地址的安全设置,可通过对活动的客户端网 ...
- Google搜索技巧-入门篇
基本搜索 Google 查询简洁方便,仅需输入查询内容并敲一下回车键 (Enter),或单击“Google 搜索”按钮即可得到相关资料. 搜索两个及两个以上关键字 Google 只会返回那些符合您的全 ...
- 学习JDK1.8集合源码之--Vector
1. Vector简介 Vector是JDK1.0版本就推出的一个类,和ArrayList一样,继承自AbstractList,实现了List.RandomAccess.Cloneable.java. ...
- Mac 电脑如何卸载 node
因为刚入手「 Mac 」很多淫技还不懂,在一次使用 npm install 的时候安装出错,提示为 npm 与 node 的版本有问题,所以就想着卸载重新装一个版本. 但是因为刚使用「 Mac 」所以 ...