YbtOJ#493-最大分数【斜率优化dp,分治】
正题
题目链接:http://www.ybtoj.com.cn/contest/117/problem/1
题目大意
\(n\)个数的一个序列,给其中的一些数打上标记。
一个标记方案的贡献为\(s_1\)表示有多少对\(L,R\)满足区间\([L,R]\)都被打上了标记,\(s_2\)表示标记的数字和。贡献为\(s_1-s_2\)
\(m\)次询问,修改一个数后,求最大的可能贡献(询问之间相互独立)
\(1\leq n,m\leq 3\times 10^5\)
解题思路
先把答案减去一个\(\sum_{i=1}^na_i\),这样就变为对于一段\([L,R]\)要么选择\(s_1\)要么选择\(s_2\)
先考虑不带修改怎么搞,设\(f_i\)表示前\(i\)个的最大贡献。
定义\(s_i=\sum_{j=1}^ia_i\)然后有方程
\]
这个东西可以斜率优化搞(考场上犯病写了个\(CDQ\),调了我半天)
然后带修改,考虑到每次只会影响一个数字,可以理解为一个前缀和后缀拼起来。把刚刚那个\(dp\)再反过来做一次记为\(g_i\)。
那么现在对于修改的位置\(x\),我们要么选择\(x\),要么跨过\(x\),也就是最大化
\]
前面那个可以把两个前/后缀的max的\(f_j-s_j\)和\(g_i+s_{i-1}\)加起来就好了。
后面那个考虑分治,我们每次分治到一个位置\([l,mid,r]\),如果需要往左走就统计\(i\in[0,l-1]\)和\(j\in[mid+1,r]\)的答案。右边同理。
因为我们的每次的时间复杂度是外面的大小,所以我们可以维护一个前后缀的凸壳,然后在上面二分时间复杂度就是\(O(n\log^2 n)\)的。
同时维护两个凸壳需要回溯所以很麻烦,分开两次正反做一次就好了。
时间复杂度\(O(n\log^2 n)\)
code
(考场代码,有点丑)
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<vector>
#define ll long long
using namespace std;
const ll N=3e5+10,inf=1e18;
ll n,m,ans,lmax,rmax,top,st[N],suf[N];
ll a[N],f[N],g[N],s[N],prt[N],w[N],yf[N];
vector<ll > q[N];
ll px(ll x,ll *f)
{return f[x]*2+x*x-x;}
ll xj(ll x1,ll y1,ll x2,ll y2)
{return x1*y2-x2*y1;}
ll xl(ll a,ll b,ll c,ll *f){
ll y1=px(b,f)-px(a,f),x1=b-a;
ll y2=px(c,f)-px(a,f),x2=c-a;
return xj(x1,y1,x2,y2);
}
ll ck(ll l,ll r,ll *f)
{return f[l]+(r-l+1)*(r-l)/2;}
ll solve(ll l,ll r,ll *f){
if(l==r)return f[l]-s[l];
ll mid=(l+r)>>1;
ll maxs=solve(l,mid,f);top=0;
for(ll i=l;i<=mid;i++){
while(top>1&&xl(st[top-1],st[top],i,f)>=0)top--;
st[++top]=i;
}
for(ll i=mid+1;i<=r;i++){
while(top>1&&ck(st[top],i,f)<ck(st[top-1],i,f))top--;
f[i]=max(f[i],max(ck(st[top],i,f),s[i]+maxs));
}
maxs=max(maxs,solve(mid+1,r,f));
return maxs;
}
ll cw(ll l,ll r)
{return f[l]+g[r]+(r-l-1)*(r-l)/2;}
ll xll(ll a,ll b,ll c){
ll y1=(yf[b]-yf[a])*2,x1=b-a;
ll y2=(yf[c]-yf[a])*2,x2=c-a;
return xj(x1,y1,x2,y2);
}
ll xlr(ll a,ll b,ll c){
ll y1=yf[b]-yf[a],x1=(n-b+1)-(n-a+1);
ll y2=yf[c]-yf[a],x2=(n-c+1)-(n-a+1);
return xj(x1,y1,x2,y2);
}
void calc(ll L,ll R){
if(L==R){
for(ll i=0;i<q[L].size();i++){
ll x=q[L][i],tmp;
prt[x]=max(ans-w[x]+a[L]-s[n],prt[x]);
}
while(top>1&&xll(st[top-1],st[top],L)>=0)top--;
st[++top]=L;
return;
}
ll mid=(L+R)>>1,pa=ans;
if(top){
for(ll i=mid+1;i<=R;i++){
ll l=1,r=top-1;
while(l<=r){
ll x=(l+r)>>1;
if(cw(st[x],i)<=cw(st[x+1],i))l=x+1;
else r=x-1;
}
ans=max(ans,cw(st[l],i));
}
}
calc(L,mid);ans=pa;
calc(mid+1,R);
return;
}
void cblc(ll L,ll R){
if(L==R){
for(ll i=0;i<q[L].size();i++){
ll x=q[L][i],tmp;
prt[x]=max(ans-w[x]+a[L]-s[n],prt[x]);
}
while(top>1&&xlr(st[top-1],st[top],L)>=0)top--;
st[++top]=L;
return;
}
ll mid=(L+R)>>1,pa=ans;
if(top){
for(ll i=L;i<=mid;i++){
ll l=1,r=top-1;
while(l<=r){
ll x=(l+r)>>1;
if(cw(i,st[x])<=cw(i,st[x+1]))l=x+1;
else r=x-1;
}
ans=max(ans,cw(i,st[l]));
}
}
cblc(mid+1,R);ans=pa;
cblc(L,mid);
return;
}
signed main()
{
freopen("score.in","r",stdin);
freopen("score.out","w",stdout);
scanf("%lld",&n);
for(ll i=1;i<=n;i++)
scanf("%lld",&a[n-i+1]);
for(ll i=1;i<=n;i++)s[i]=s[i-1]+a[i];
solve(0,n,g);
for(ll i=1;i<n-i+1;i++)
swap(a[i],a[n-i+1]),swap(g[i],g[n-i+1]);
for(ll i=1;i<=n;i++)s[i]=s[i-1]+a[i];
solve(0,n,f);
scanf("%lld",&m);
for(ll i=1;i<=m;i++){
ll x;
scanf("%lld%lld",&x,&w[i]);
q[x].push_back(i);
}
suf[n+2]=-inf;
for(ll i=n+1;i>=1;i--)
suf[i]=max(suf[i+1],g[i]+s[i-1]);
for(ll i=0,pre=-inf;i<=n;i++){
for(ll j=0;j<q[i].size();j++){
ll x=q[i][j];
prt[x]=suf[i+1]+pre-s[n];
}
pre=max(pre,f[i]-s[i]);
}
ans=-inf;top=0;
for(ll i=1;i<=n;i++)yf[i]=f[i]*2+i*i+i;calc(0,n+1);
ans=-inf;top=0;
for(ll i=1;i<=n;i++)yf[i]=g[i]*2+(n-i+1)*(n-i+1)+(n-i+1);
cblc(0,n+1);
for(ll i=1;i<=m;i++)
printf("%lld\n",max(prt[i],0ll));
return 0;
}
YbtOJ#493-最大分数【斜率优化dp,分治】的更多相关文章
- bzoj2149拆迁队 斜率优化dp+分治
2149: 拆迁队 Time Limit: 10 Sec Memory Limit: 259 MBSubmit: 397 Solved: 177[Submit][Status][Discuss] ...
- 【BZOJ-1492】货币兑换Cash DP + 斜率优化 + CDQ分治
1492: [NOI2007]货币兑换Cash Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 3396 Solved: 1434[Submit][Sta ...
- 【uoj#244】[UER #7]短路 CDQ分治+斜率优化dp
题目描述 给出 $(2n+1)\times (2n+1)$ 个点,点 $(i,j)$ 的权值为 $a[max(|i-n-1|,|j-n-1|)]$ ,找一条从 $(1,1)$ 走到 $(2n+1,2n ...
- 【bzoj3672】[Noi2014]购票 斜率优化dp+CDQ分治+树的点分治
题目描述 给出一棵以1为根的带边权有根树,对于每个根节点以外的点$v$,如果它与其某个祖先$a$的距离$d$不超过$l_v$,则可以花费$p_vd+q_v$的代价从$v$到$a$.问从每个点到1花费 ...
- BZOJ1492:[NOI2007]货币兑换 (CDQ分治+斜率优化DP | splay动态维护凸包)
BZOJ1492:[NOI2007]货币兑换 题目传送门 [问题描述] 小Y最近在一家金券交易所工作.该金券交易所只发行交易两种金券:A纪念券(以下简称A券)和B纪念券(以下简称B券).每个持有金券的 ...
- 2018.09.29 bzoj3675: [Apio2014]序列分割(斜率优化dp)
传送门 斜率优化dp经典题目. 首先需要证明只要选择的K个断点是相同的,那么得到的答案也是相同的. 根据分治的思想,我们只需要证明有两个断点时成立,就能推出K个断点时成立. 我们设两个断点分成的三段连 ...
- [Noi2014]购票 斜率优化DP+可持久化凸包
貌似网上大部分题解都是CDQ分治+点分治然后再斜率优化DP,我貌似并没有用这个方法. 这一题跟这题有点像,只不过多了一个l的限制 如果说直接跑斜率优化DP,存储整个序列的话,显然是不行的,如图所示(图 ...
- 斜率优化DP学习笔记
先摆上学习的文章: orzzz:斜率优化dp学习 Accept:斜率优化DP 感谢dalao们的讲解,还是十分清晰的 斜率优化$DP$的本质是,通过转移的一些性质,避免枚举地得到最优转移 经典题:HD ...
- 【bzoj3675】[Apio2014]序列分割 斜率优化dp
原文地址:http://www.cnblogs.com/GXZlegend/p/6835179.html 题目描述 小H最近迷上了一个分隔序列的游戏.在这个游戏里,小H需要将一个长度为n的非负整数序列 ...
随机推荐
- .net 的析构函数和dispose模式
- 关于Ubuntu18.04 linux系统使用安装JDK Mysql
平台部署 一.安装JDK step1.下载OracleJDKstep2. 解压step3. 加入环境变量 具体操作如下: lemon@ubuntu:~$ cd ~/download/ lemon@ub ...
- hive -- 外部表、内部表、临时表
1.外部表 关键字:EXTERNAL 外部表创建时需要指定LOCATION 删除外部表时,数据不被删除 CREATE EXTERNAL TABLE page_view(viewTime INT, us ...
- Java String.split()的特殊用法
1 //用多种字符分隔字符串 2 public class Main { 3 /* 4 * "(1,2),(2,4),(3,6),(4,7)"按[(),]分隔 5 * 空白(1,2 ...
- 网络操作系统VyOS之NAT实践
本文基于 网络操作系统VyOS应用实践(四) 修改,完善了实验细节及1-to-1 NAT部分. NAT NAT即网络地址转换,最常见的就是各种虚拟机工具的NAT模式,让虚拟机以宿主的网络地址与外网通讯 ...
- 跨域@RequestBody@RequestParam 和JSON.stringify
- JOB状态与并发
由于job每次被执行时都会创建一个新的实例, jobDetail实例时,要进行数据存储或者,特殊字段操作,需要每次schedul执行job时保留之前的数据, 那么就需要job在有状态下保持之前的数据信 ...
- MySQL 事务和锁
1. 事务 1.1 什么是事务? 1.2 事务的特性:ACID 1.3 事务语句 1.4 事务的隔离级别 1.5 锁 1.6 事务隔离解决并发问题 2. 死锁 2.1 场景示例 2.2 死锁调优 3. ...
- Java同步之线程池详解
带着问题阅读 1.什么是池化,池化能带来什么好处 2.如何设计一个资源池 3.Java的线程池如何使用,Java提供了哪些内置线程池 4.线程池使用有哪些注意事项 池化技术 池化思想介绍 池化思想是将 ...
- Stream流用于按照对象中某一属性来对集合去重+简单数据类型集合的去重
上次对Stream流来进行分组的文章很多人看,想看的可以来这: Stream流来进行集合分组 这次小编又带来Stream的去重,话不多数,直接上代码: 这是对简单数据类型的去重 //字符串集合进行简单 ...