【斜率优化】bzoj3675-[Apio2014]序列分割&&Uoj104
题目大意
将一个长度为N的非负整数序列分割成k+l个非空的子序列,每次选择一位置分割后,将会得到一定的分数,这个分数为两个新序列中元素和的乘积。求最大的分数。
[UOJ104]并输出任意一种方案
思路
显然,无论分割顺序如何,不会影响最后得到的结果。所以可以利用递推方程。\(f[i][j]\)表示取前\(i\)个数,分割成\(j\)个序列能得到的最大分数。显然有:
\]
当\(Ans_{j_1}>Ans_{j_2}\)时,有:
\]
\]
\]
令\(x[i]=f[i][k-1]-sum[i]^2,y[i]=sum[i]\)
则有:
\]
注意点
计算斜率的时候\(x_1\)可能等于\(x_2\),特判一下将斜率设为INF或-INF。不要忘记开long long。
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef long long LL;
const int MAXN=100100;
const int MAXK=250;
int n,k;
LL sum[MAXN],x[2][MAXN],g[MAXN],y[MAXN],f[MAXN][2];
int cur;
void init()
{
scanf("%d%d",&n,&k);
for (int i=1;i<=n;i++)
{
int tmp;
scanf("%d",&tmp);
y[i]=sum[i]=sum[i-1]+tmp;
g[i]=-y[i];
}
}
LL dp()
{
memset(f,0,sizeof(f));
cur=0;
for (int i=1;i<=n;i++) x[1-cur][i]=f[i][cur]-(sum[i]*sum[i]);
for (int j=2;j<=k+1;j++)
{
cur=1-cur;
int head=0,tail=1,que[MAXN];
for (int i=j-1;i<=n;i++)//上一次至多分割为j-1部分,则至少从j-1开始
{
while (head+1<tail && x[cur][que[head]]-x[cur][que[head+1]]<=g[i]*(y[que[head]]-y[que[head+1]])) head++;
int best=que[head];
f[i][cur]=f[best][1-cur]+sum[best]*(sum[i]-sum[best]);
while (head+1<tail && (LL)(x[cur][que[tail-1]]-x[cur][i])*(y[que[tail-2]]-y[que[tail-1]])>=(LL)(x[cur][que[tail-2]]-x[cur][que[tail-1]])*(y[que[tail-1]]-y[i])) tail--;
que[tail++]=i;
x[1-cur][i]=f[i][cur]-(sum[i]*sum[i]);
}
}
return (f[n][cur]);
}
void printans()
{
printf("%lld\n",dp());
}
int main()
{
init();
printans();
return 0;
}
输出方案
只需记录一下路径就好了。不过要注意,UOJ后面数据时间卡得非常可啪,所以我们不用斜率而是直接用乘法来计算,同时x数组y数组g数组也不要了直接套进去算,勉勉强强卡过了……[痛心疾首.jpg]
顺带一提的是,这样的话BZOJ会T(咦?)
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef long long LL;
const int MAXN=100100;
const int MAXK=250;
int n,k;
LL sum[MAXN],f[MAXN][2],fr[MAXN][MAXK];
int cur;
void init()
{
scanf("%d%d",&n,&k);
for (int i=1;i<=n;i++)
{
int tmp;
scanf("%d",&tmp);
sum[i]=sum[i-1]+tmp;
}
}
LL x(int m)
{
return f[m][1-cur]-sum[m]*sum[m];
}
LL dp()
{
memset(f,0,sizeof(f));
cur=0;
for (int j=2;j<=k+1;j++)
{
cur=1-cur;
int head=0,tail=1,que[MAXN];
for (int i=j-1;i<=n;i++)//上一次至多分割为j-1部分,则至少从j-1开始
{
while (head+1<tail && (f[que[head]][1-cur]-(LL)(sum[que[head]]*sum[que[head]])-f[que[head+1]][1-cur]+(LL)(sum[que[head+1]]*sum[que[head+1]])<=(LL)-sum[i]*(sum[que[head]]-sum[que[head+1]]))) head++;
int best=que[head];
f[i][cur]=f[best][1-cur]+sum[best]*(sum[i]-sum[best]);
fr[i][j]=best;
while (head+1<tail && (LL)(x(que[tail-1])-x(i))*(sum[que[tail-2]]-sum[que[tail-1]])>=(LL)(x(que[tail-2])-x(que[tail-1]))*(sum[que[tail-1]]-sum[i])) tail--;
que[tail++]=i;
}
}
return (f[n][cur]);
}
void printans()
{
printf("%lld\n",dp());
int ans[MAXK];
memset(ans,0,sizeof(ans));//不要忘记初始化★
for (int i=k+1;i>=2;i--)
{
ans[++ans[0]]=fr[n][i];
n=fr[n][i];
}
for (int i=ans[0];i>=1;i--) printf("%d ",ans[i]);
}
int main()
{
init();
printans();
return 0;
}
【斜率优化】bzoj3675-[Apio2014]序列分割&&Uoj104的更多相关文章
- bzoj3675[Apio2014]序列分割 斜率优化dp
3675: [Apio2014]序列分割 Time Limit: 40 Sec Memory Limit: 128 MBSubmit: 3508 Solved: 1402[Submit][Stat ...
- [Bzoj3675][Apio2014]序列分割(斜率优化)
3675: [Apio2014]序列分割 Time Limit: 40 Sec Memory Limit: 128 MBSubmit: 4021 Solved: 1569[Submit][Stat ...
- BZOJ3675 [Apio2014]序列分割 【斜率优化dp】
3675: [Apio2014]序列分割 Time Limit: 40 Sec Memory Limit: 128 MB Submit: 3366 Solved: 1355 [Submit][St ...
- BZOJ3675 [Apio2014]序列分割 动态规划 斜率优化
原文链接http://www.cnblogs.com/zhouzhendong/p/8697258.html 题目传送门 - BZOJ3675 题意 对于一个非负整数序列,小H需要重复k次以下的步骤: ...
- BZOJ3675: [Apio2014]序列分割(斜率优化)
Time Limit: 40 Sec Memory Limit: 128 MBSubmit: 4186 Solved: 1629[Submit][Status][Discuss] Descript ...
- 2018.09.29 bzoj3675: [Apio2014]序列分割(斜率优化dp)
传送门 斜率优化dp经典题目. 首先需要证明只要选择的K个断点是相同的,那么得到的答案也是相同的. 根据分治的思想,我们只需要证明有两个断点时成立,就能推出K个断点时成立. 我们设两个断点分成的三段连 ...
- BZOJ3675 Apio2014 序列分割 【斜率优化】
Description 小H最近迷上了一个分隔序列的游戏.在这个游戏里,小H需要将一个长度为n的非负整数序列分割成k+1个非空的子序列.为了得到k+1个子序列,小H需要重复k次以下的步骤: 1.小H首 ...
- [luogu3648][bzoj3675][APIO2014]序列分割【动态规划+斜率优化】
题目大意 让你把一个数列分成k+1个部分,使分成乘积分成各个段乘积和最大. 分析 首先肯定是无法开下n \(\times\) n的数组,那么来一个小技巧:因为我们知道k的状态肯定是从k-1的状态转移过 ...
- bzoj3675: [Apio2014]序列分割
留坑 为什么别人家的斜率优化跟我一点都不一样! 为什么斜率都要变成正的... 为什么要那么推式子 为什么不能直接做啊..... 为什么不把0去掉去秒WA啊 为什么叉积去了0也过不了啊 woc啊 #in ...
随机推荐
- E题hdu 1425 sort
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1425 sort Time Limit: 6000/1000 MS (Java/Others) M ...
- 控制 Cookie 的作用范围
默认时,网站的所有 Cookies 都一起被存储在客户端,并且所有 Cookies 连同网站的任何请求一起被发送到服务器.换句话说,网站中的每个页面都能够为网站获取所有的 Cookies.但是,你能够 ...
- mysql in/no in/like
% 任意字符 _ 任意一个字符 in (value,......) 在这里 not in (value,......) 不在这里 mysql> select 'a' not in (1,2,3, ...
- ES6 新增的一些东西
一.常量 不允许重复定义 const a='HELLO' const a='world'//报错Uncaught SyntaxError: Identifier 'a' has already bee ...
- mongodb-linux-x86_64
卷 DataDisk 的文件夹 PATH 列表卷序列号为 4A8E-D95CD:.│ 1.txt│ GNU-AGPL-3.0│ MPL-2│ README│ THIRD-PARTY-NOTI ...
- 【bzoj1798】【AHOI2009】维护序列
练一下线段树模板,区间乘法. #include<bits/stdc++.h> #define lson (o<<1) #define rson (o<<1|1) ; ...
- python的多线程、多进程代码示例
python多进程和多线程的区别:python的多线程不是真正意义上的多线程,由于python编译器的问题,导致python的多线程存在一个PIL锁,使得python的多线程的CPU利用率比预期的要低 ...
- 这种文件别打开, 大小不足1KB, 却可以让你电脑瘫痪
今年6月份,抖音表白代码火了,不足1kb的txt文件,玩出了新花样.可是你知道吗,这种非常“浪漫”的表白方式,其实存在着很大的风险,甚至会让你的电脑直接瘫痪. 首先,先说一下所谓的表白代码是怎么回事. ...
- LR运行场景时出现的error
LR运行场景时出现的error 1.Action.c(24): Error -27740: Overlapped transmission of request to "home.asiai ...
- bzoj 1477 扩展欧几里德
思路:很裸的求相遇问题. #include<bits/stdc++.h> #define LL long long #define fi first #define se second # ...