【斜率优化】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 ...
随机推荐
- 用例图(Use Case Diagram)
用例图(Use Case Diagram) 执行者/参与者(Actor): 表示与您的应用程序或系统进行交互的用户.组织或外部系统.用一个小人表示. 用例(Use Case): 即系统具有的功能,在用 ...
- Fiddler-- 安装HTTPs证书
1. 现在很多带有比较重要信息的接口都使用了安全性更高的HTTPS,而Fiddler默认是抓取HTTP类型的接口,要想查看HTTPS类型接口就需要安装fiddler证书. 2.打开Fiddler, ...
- hdfs文件上传机制与namenode元数据管理机制
1.hdfs文件上传机制 文件上传过程: 1.客户端想NameNode申请上传文件, 2.NameNode返回此次上传的分配DataNode情况给客户端 3.客户端开始依向dataName上传对应 ...
- 【UOJ224】短路
具体可以看UOJmyy的blog,orz 就是一个贪心. #include<bits/stdc++.h> typedef long long ll; using namespace std ...
- NOIP 2012 Day1
tags: NOIP 模拟 倍增 高精 Python categories: 信息学竞赛 总结 Luogu P1079 Vigenère 密码 Solution 表示并不是很懂其他人发的题解. 我是这 ...
- java-增强for循环
public static void main(String[] args) { ArrayList<String> list = new ArrayList<>(); lis ...
- intellij idea 无法创建类文件,方法。
- LightOJ 1282
Leading and Trailing Time Limit: 2000MS Memory Limit: 32768KB 64bit IO Format: %lld & %llu S ...
- python和语法糖
语法糖(syntactic sugar)是指编程语言中可以更容易的表达一个操作的语法,它可以使程序员更加容易去使用这门语言:操作可以变得更加清晰.方便,或者更加符合程序员的编程习惯.(百度百科的 ...
- LeetCode解题报告—— Longest Valid Parentheses
Given a string containing just the characters '(' and ')', find the length of the longest valid (wel ...