@description@

给定序列 T1, T2, ... TN,你可以从中选择一些 Ti,可以选择 0 个(即不选)。

定义你选择的权值 = (满足 T[L...R] 都被选择的区间 [L, R] 的数量)-(你选择的 Ti 之和),你希望这个权值尽量大。

现在有 M 次询问,每次询问假如将 T[Pi] 修改成 Xi,你所能选出的最大权值。

Constraints

1 <= N, M <= 3*10^5,1 <= Ti <= 10^9 且所有 Ti 总和 <= 10^12。

对于每组询问,满足 1 <= Pi <= N,1 <= Xi <= 10^9。

Input

输入形式如下:

N

T1 T2 … TN

M

P1 X1

P2 X2

:

PM XM

Output

对于每组询问,输出其答案。

Sample Input 1

5

1 1 4 1 1

2

3 2

3 10

Sample Output 1

9

2

@solution@

考虑在修改 T[Pi] 为 Xi 后,最后的选择方案要么包含 Pi 要么不包含。

假如不包含,则只需要在 1 ... Pi - 1 与 Pi + 1 ... N 中选择得到最大值。

假如包含,我们需要预处理包含 Pi 的最优解,然后用这个最优解 + T[Pi] - Xi。

综上,我们需要处理出三个东西:第一个 f[i] 表示从 1 ... i 中选择的最优答案,第二个 g[i] 表示从 i ... N 中选择的最优答案,第三个 h[i] 表示必须选择 i 的最优答案。

先一步步考虑,考虑求 f[i]。要么不选 i 从 f[i-1] 转移过来;要么选择 i,我们枚举一个 j 表示我们强制选择 i + 1 ... j 中的所有数,则贡献为 f[j] + ([i + 1, j] 中的区间数量)-([i + 1, j] 的 T 值之和)。

通过预处理前缀和 S[i] = T[1] + T[2] + ... T[i],可以将转移式写得更具体一点:

\[f[i] = \max(f[i-1], \max_{0\le j<i}(f[j] + S[j] - S[i] + \frac{(i-j+1)*(i-j)}{2}))
\]

上面那个转移非常斜率优化,稍微变化一下式子发现的确可以斜率优化。这里就不具体写了。

总之,这个式子是个斜率单增且横坐标单增的斜率优化,可以使用单调栈做到 O(n) 的时间复杂度。

而 g[i] 可以类似地处理。这里也不具体谈了。

现在考虑怎么求 h[i]。最暴力的方法是在 i 左边枚举一个左端点 l,i 右边枚举一个右端点 r,算出区间 [l, r] 的贡献再加上 f[l - 1] + g[r + 1]。

考虑优化,我们发现 cdq 分治非常合适(其实我也不知道它该不该叫作cdq分治,只是感觉比较像)。

对于区间 [left, right],设它的中点为 mid,我们仅考虑满足 left <= l <= mid < r <= right 的区间 [l, r] 对 h[l...r] 的贡献。

然后递归到 [left, mid] 与 [mid + 1, right],继续处理子问题,直到 left = right。

怎么处理呢?我们先将 [left, mid] 的点加入单调栈;然后从左往右扫 [mid + 1, right] 用斜率优化求出 [mid + 1, right] 内所有点作为右端点的答案;然后再从右往左扫描,扫到 x 时维护出 [x, right] 的最大值 tmp,用这个 tmp 更新 h[x]。

再反过来将 [mid + 1, right] 加入单调栈,更新 [left, mid] 的 h 值。这个过程与上面类似,不再赘述。

@accepted code@

#include<cstdio>
#include<algorithm>
using namespace std;
typedef long long ll;
const int MAXN = 300000;
const ll INF = (1LL<<60);
ll f[MAXN + 5], g[MAXN + 5], h[MAXN + 5];
ll S[MAXN + 5], T[MAXN + 5];
ll fk(int i) {return i;}
ll fx(int j) {return 2LL*j;}
ll fc(int i) {return 1LL*i*i + i - 2*S[i];}
ll fy(int j) {return 2*S[j] + f[j] - j + 1LL*j*j;}
double fslope(int p, int q) {return 1.0*(fy(q) - fy(p)) / (fx(q) - fx(p));}
ll gk(int i) {return -i;}
ll gx(int j) {return -2LL*j;}
ll gc(int i) {return 1LL*i*i - i + 2*S[i-1];}
ll gy(int j) {return -2*S[j-1] + g[j] + j + 1LL*j*j;}
double gslope(int p, int q) {return 1.0*(gy(q) - gy(p)) / (gx(q) - gx(p));}
int stk[MAXN + 5], tp;
int N, M;
void get_dp() {
f[0] = 0; stk[tp = 1] = 0;
for(int i=1;i<=N;i++) {
while( tp > 1 && fk(i) >= fslope(stk[tp], stk[tp-1]) )
tp--;
f[i] = max(f[i - 1], fc(i) - fk(i)*fx(stk[tp]) + fy(stk[tp]));
while( tp > 1 && fslope(i, stk[tp]) >= fslope(stk[tp], stk[tp-1]) )
tp--;
stk[++tp] = i;
}
g[N + 1] = 0; stk[tp = 1] = N + 1;
for(int i=N;i>=1;i--) {
while( tp > 1 && gk(i) >= gslope(stk[tp], stk[tp-1]) )
tp--;
g[i] = max(g[i + 1], gc(i) - gk(i)*gx(stk[tp]) + gy(stk[tp]));
while( tp > 1 && gslope(i, stk[tp]) >= gslope(stk[tp], stk[tp-1]) )
tp--;
stk[++tp] = i;
}
}
ll tmp[MAXN + 5];
void divide_conquer(int l, int r) {
if( l == r ) {
h[l] = f[l + 1] - 2*T[l] + g[r + 1] + 2;
return ;
}
int mid = (l + r) >> 1;
divide_conquer(l, mid);
divide_conquer(mid + 1, r);
stk[tp = 1] = l - 1;
for(int i=l;i<mid;i++) {
while( tp > 1 && fk(i) >= fslope(stk[tp], stk[tp-1]) )
tp--;
while( tp > 1 && fslope(i, stk[tp]) >= fslope(stk[tp], stk[tp-1]) )
tp--;
stk[++tp] = i;
}
for(int i=mid+1;i<=r;i++) {
while( tp > 1 && fk(i) >= fslope(stk[tp], stk[tp-1]) )
tp--;
tmp[i] = fc(i) - fk(i)*fx(stk[tp]) + fy(stk[tp]) + g[i + 1];
}
ll p = -INF;
for(int i=r;i>mid;i--) {
p = max(p, tmp[i]);
h[i] = max(h[i], p);
}
stk[tp = 1] = r + 1;
for(int i=r;i>mid+1;i--) {
while( tp > 1 && gk(i) >= gslope(stk[tp], stk[tp-1]) )
tp--;
while( tp > 1 && gslope(i, stk[tp]) >= gslope(stk[tp], stk[tp-1]) )
tp--;
stk[++tp] = i;
}
for(int i=mid;i>=l;i--) {
while( tp > 1 && gk(i) >= gslope(stk[tp], stk[tp-1]) )
tp--;
tmp[i] = gc(i) - gk(i)*gx(stk[tp]) + gy(stk[tp]) + f[i - 1];
}
p = -INF;
for(int i=l;i<=mid;i++) {
p = max(p, tmp[i]);
h[i] = max(h[i], p);
}
}
int main() {
scanf("%d", &N);
for(int i=1;i<=N;i++)
scanf("%lld", &T[i]), S[i] = S[i - 1] + T[i];
get_dp(), divide_conquer(1, N);
scanf("%d", &M);
for(int i=1;i<=M;i++) {
int P, X; scanf("%d%d", &P, &X);
printf("%lld\n", max(f[P-1] + g[P+1], h[P] + 2*T[P] - 2*X)/2);
}
}

@details@

h[i] 值是必须要包含 i 的,所以可能为负数,给 h 初始化时应该赋值为 -inf。

话说现在 atcoder 都不办 ARC 了,ARC 已经成为了时代的眼泪。。。

@atcoder - ARC066F@ Contest with Drinks Hard的更多相关文章

  1. [arc066f]Contest with Drinks Hard

    题目大意: 有一些物品,每个买了有代价. 如果存在一个极大区间[l,r]内的物品都被买了,这个区间长度为k,可以获得的收益是k*(k+1)/2. 现在若干次询问,每次问假如修改了某个物品的价格,最大收 ...

  2. AtCoder Beginner Contest 050 ABC题

    A - Addition and Subtraction Easy Time limit : 2sec / Memory limit : 256MB Score : 100 points Proble ...

  3. AtCoder Regular Contest 061

    AtCoder Regular Contest 061 C.Many Formulas 题意 给长度不超过\(10\)且由\(0\)到\(9\)数字组成的串S. 可以在两数字间放\(+\)号. 求所有 ...

  4. AtCoder Grand Contest 012

    AtCoder Grand Contest 012 A - AtCoder Group Contest 翻译 有\(3n\)个人,每一个人有一个强大值(看我的假翻译),每三个人可以分成一组,一组的强大 ...

  5. AtCoder Regular Contest 094 (ARC094) CDE题解

    原文链接http://www.cnblogs.com/zhouzhendong/p/8735114.html $AtCoder\ Regular\ Contest\ 094(ARC094)\ CDE$ ...

  6. AtCoder Grand Contest 011

    AtCoder Grand Contest 011 upd:这篇咕了好久,前面几题是三周以前写的... AtCoder Grand Contest 011 A - Airport Bus 翻译 有\( ...

  7. AtCoder Grand Contest 031 简要题解

    AtCoder Grand Contest 031 Atcoder A - Colorful Subsequence description 求\(s\)中本质不同子序列的个数模\(10^9+7\). ...

  8. AtCoder Grand Contest 010

    AtCoder Grand Contest 010 A - Addition 翻译 黑板上写了\(n\)个正整数,每次会擦去两个奇偶性相同的数,然后把他们的和写会到黑板上,问最终能否只剩下一个数. 题 ...

  9. AtCoder Grand Contest 009

    AtCoder Grand Contest 009 A - Multiple Array 翻译 见洛谷 题解 从后往前考虑. #include<iostream> #include< ...

随机推荐

  1. 使用Jedis操作Redis-使用Java语言在客户端操作---List类型

    在Redis中,List类型是按照插入顺序排序的字符串链表.和数据结构中的普通链表一样,我们可以在其头部(left)和尾部(right)添加新的元素.在插入时,如果该键并不存在,Redis将为该键创建 ...

  2. 直接在安装了redis的Linux机器上操作redis数据存储类型--String类型

    一.概述: 字符串类型是Redis中最为基础的数据存储类型,它在Redis中是二进制安全的,这便意味着该类型可以接受任何格式的数据,如JPEG图像数据或Json对象描述信息等.在Redis中字符串类型 ...

  3. 遍历list时删除元素的正确做法

    我们往往会遇到需要删除list中满足条件的元素.举例: List<string> list_str =new List<string>() { "A",&q ...

  4. Ubuntu 16.04 LTS安装Docker最新版

    一.安装Docker的先决条件 1.运行64位CPU构架的计算机(目前只能是x86_64和amd64),请注意,Docker目前不支持32位CPU.2.运行Linux 3.8或更高版本内核.一些老版本 ...

  5. mysql 查询条件不区分大小写问题

    转自 http://blog.csdn.net/qishuo_java/article/details/40118937 转自 https://www.cnblogs.com/wuyun-blog/p ...

  6. 洛谷 P1217 [USACO1.5]回文质数 Prime Palindromes【取回文数/数论/字符串】

    题目描述 因为151既是一个质数又是一个回文数(从左到右和从右到左是看一样的),所以 151 是回文质数. 写一个程序来找出范围[a,b](5 <= a < b <= 100,000 ...

  7. 禁用/移除WordPress页面的评论功能

    对于某些类型的WordPress站点,也许不需要在页面(page)提供评论功能,那么你可以通过下面的方法,很容易就禁用或移除WordPress页面的评论功能. 方法1:在页面编辑界面取消该页面的评论功 ...

  8. tyvj 1864 守卫者的挑战

    传送门 解题思路 dp[i][j][k]表示前i个挑战,赢了j场,现在还有k个包的获胜概率. 转移方程: dp[i+1][j+1][k+a[i]] += p[i+1]*dp[i][j][k] (k+a ...

  9. org.apache.jasper.JasperException: xxxx.jsp(118,24) Attribute style invalid for tag formatNumber according to TLD

    错误:org.apache.jasper.JasperException: /projm/projBudgetChangeOverview.jsp(118,24) Attribute style in ...

  10. Django框架Day3------之Models

    一.Django models字段类型清单: AutoField:一个自动递增的整型字段,添加记录时它会自动增长.你通常不需要直接使用这个字段:如果你不指定主键的话,系统会自动添加一个主键字段到你的m ...