bzoj-3675 序列分割

题意:

给出一个长度为n的序列和m次操作;

每次操作选择一个地方将序列分开,得分为左一半的和乘右一半;

求最大的总得分;

题解:

首先显然分割的顺序与得分无关,只要在那几个地方割就可以是最优解;

(为什么?难道要我@PoPoQQQ来证明一下?)

然后可以设状态为f[i][j],表示在前j个数里选择i个点分割的最大得分;

那么转移式显然为f[i][j]=max(f[i-1][k]+(s[n]-s[j])*(s[j]-s[k]))

(0<=k<j,s[x]表示x的前缀和)

O(n^2)的复杂度,所以要优化一下;

对于i的更新,考虑对于两个决策k,j  (k<j),当j优于k时;

f[i-1][k]+(s[n]-s[i])*(s[i]-s[k])<f[i-1][j]+(s[n]-s[i])*(s[i]-s[j]);

f[i-1][k]-(s[n]-s[i])*s[k]<f[i-1][j]-(s[n]-s[i])*s[j];

f[i-1][k]-f[i-1][j]<(s[n]-s[i])*(s[k]-s[j]);

(f[i-1][k]-f[i-1][j])/(s[k]-s[j])>s[n]-s[i];

对于i来说,只要满足此式的k,j,那么j就优于k;

并且因为右面是关于i不增的,所以只要有一个i满足之后,从此k就永远不会比j优;

所以此时可以将k删掉,也就是说可以维护一个单调队列来优化这个式子了;

但是这个是有可能维护出的不是最优解的情况;

先定义(f[i-1][k]-f[i-1][j])/(s[k]-s[j])为斜率;

对于三个决策k,j,i;

倘若斜率(k,j)>s[n]-s[i]并且斜率(j,i)也满足的话,是无法对k与i比较,得出更优的解的;

所以分别讨论斜率(k,j)与斜率(j,i)与s[n]-s[i]的关系,就可以得出这个斜率只能的递减的;

因为递增时j可以被删掉;

这时就可以在O(n)的复杂度内解决问题了;

HINT:

1.OJ卡了一下MLE,数组滚动一下就好;

2.有各种负数在不等号两边摇摆,注意不等号变向;

3.long long

4.其实我写这篇题解就是为了复习一下斜率优化...然而最后一段懒得写了;

具体可以移步blog另一个斜率优化...

代码:

#include<stdio.h>
#include<string.h>
#include<algorithm>
#define N 100001
using namespace std;
typedef long long ll;
ll a[N],s[N],f[2][N];
int q[N];
int main()
{
	int n,m,i,j,k,st,en;
	ll ans;
	scanf("%d%d",&n,&m);
	for(i=1;i<=n;i++)
		scanf("%lld",a+i),s[i]=s[i-1]+a[i];
	for(i=1;i<=m;i++)
	{
		q[st=en=1]=0;
		for(j=1;j<n;j++)
		{
			while(st<en&&f[(i-1)%2][q[st]]-f[(i-1)%2][q[st+1]]<=(s[n]-s[j])*(s[q[st]]-s[q[st+1]]))
				st++;
			f[i%2][j]=f[(i-1)%2][q[st]]+(s[n]-s[j])*(s[j]-s[q[st]]);
			while(st<en&&(f[(i-1)%2][q[en]]-f[(i-1)%2][q[en-1]])*(s[q[en]]-s[j])>=(f[(i-1)%2][q[en]]-f[(i-1)%2][j])*(s[q[en]]-s[q[en-1]]))
				en--;
			q[++en]=j;
		}
	}
	for(i=1,ans=0;i<n;i++)
		ans=max(ans,f[m%2][i]);
	printf("%lld",ans);
	return 0;
}

时间: 2024-10-10 22:06:37

bzoj-3675 序列分割的相关文章

【BZOJ】【3675】【APIO2014】序列分割

DP+斜率优化 首先我们根据这个分割的过程可以发现:总得分等于k+1段两两的乘积的和(乘法分配律),也就是说与分割顺序是无关的. 再对乘积进行重分组(还是乘法分配律)我们可以转化为:$ans=\sum$第 i 段×前 i-1 段的和 所以我们就可以以分割次数为阶段进行DP啦- 令f[i][j]表示将前 j 个数分成 i 段的最大得分,那么就有$$f[i][j]=max\{ f[i-1][k]+sum[k]×(sum[j]-sum[k]) \}$$我们观察到这个式子其实是很像斜率优化的……而且su

bzoj 3675 [Apio2014]序列分割(斜率DP)

[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=3675 [题意] 将n个数的序列分割k次,每次的利益为分割后两部分数值和的积,求最大利益. [思路] 设f[i][j]表示将前i个分割j次的最大获益,则有转移式: f[i][j]=max{ f[k][j-1]+(S(i)-S(k))*S(k) } 设a<b,若b决策优于a决策则有: (S[b]^2-S[a]^2+f[a][j-1]-f[b][j-1])/(S[b]-S[a])<S[i

【斜率DP】BZOJ 3675:[Apio2014]序列分割

3675: [Apio2014]序列分割 Time Limit: 40 Sec  Memory Limit: 128 MBSubmit: 1066  Solved: 427[Submit][Status][Discuss] Description 小H最近迷上了一个分割序列的游戏.在这个游戏里,小H需要将一个长 度为N的非负整数序列分割成k+l个非空的子序列.为了得到k+l个子序列, 小H将重复进行七次以下的步骤: 1.小H首先选择一个长度超过1的序列(一开始小H只有一个长度为n的 序列一一也就

【bzoj 3675】[Apio2014]序列分割

Description 小H最近迷上了一个分隔序列的游戏.在这个游戏里,小H需要将一个长度为n的非负整数序列分割成k+1个非空的子序列.为了得到k+1个子序列,小H需要重复k次以下的步骤: 1.小H首先选择一个长度超过1的序列(一开始小H只有一个长度为n的序列--也就是一开始得到的整个序列): 2.选择一个位置,并通过这个位置将这个序列分割成连续的两个非空的新序列. 每次进行上述步骤之后,小H将会得到一定的分数.这个分数为两个新序列中元素和的乘积.小H希望选择一种最佳的分割方式,使得k轮之后,小

动态规划(斜率优化):BZOJ 3675 [Apio2014]序列分割

Description 小H最近迷上了一个分割序列的游戏.在这个游戏里,小H需要将一个长 度为N的非负整数序列分割成k+l个非空的子序列.为了得到k+l个子序列, 小H将重复进行七次以下的步骤: 1.小H首先选择一个长度超过1的序列(一开始小H只有一个长度为n的 序列一一也就是一开始得到的整个序列): 2.选择一个位置,并通过这个位置将这个序列分割成连续的两个非空的新 序列. 每次进行上述步骤之后,小H将会得到一定的分数.这个分数为两个新序 列中元素和的乘积.小H希望选择一种最佳的分割方案,使得

BZOJ 3675 APIO2014 序列分割 斜率优化

题目大意:给定一个序列,可以分割k次,每次分割的得分为两段序列的和的乘积 求最大得分 首先我们可以推出序列的分割顺序是不影响得分的 比如说我要把一个序列分割成四份ABCD 我先分割A BCD或者先分割AB CD最后的得分是一样的 证明?嗯--易证.显然嘛.哈哈.好吧我不会证...自己画一下推推就好 好吧这是神犇的证法:比如我将ABCD分割为AB CD 那么A就和CD各乘了一次 B也和CD各乘了一次 再分割AB时A和B也乘了一次 最后可以保证所有的序列对(X,Y)在任何一种分割法中都只乘了一次 然

3675: [Apio2014]序列分割

Time Limit: 40 Sec  Memory Limit: 128 MBSubmit: 4522  Solved: 1752[Submit][Status][Discuss] Description 小H最近迷上了一个分隔序列的游戏.在这个游戏里,小H需要将一个长度为n的非负整数序列分割成k+1个非空的子序列.为了得到k+1个子序列,小H需要重复k次以下的步骤: 1.小H首先选择一个长度超过1的序列(一开始小H只有一个长度为n的序列——也就是一开始得到的整个序列): 2.选择一个位置,并

[BZOJ3675]序列分割

3675: [Apio2014]序列分割 Time Limit: 40 Sec  Memory Limit: 128 MB Description 小H最近迷上了一个分隔序列的游戏.在这个游戏里,小H需要将一个长度为n的非负整数序列分割成k+1个非空的子序列.为了得到k+1个子序列,小H需要重复k次以下的步骤: 1.小H首先选择一个长度超过1的序列(一开始小H只有一个长度为n的序列--也就是一开始得到的整个序列): 2.选择一个位置,并通过这个位置将这个序列分割成连续的两个非空的新序列. 每次进

bzoj3675【APIO2014】序列分割

3675: [Apio2014]序列分割 Time Limit: 40 Sec  Memory Limit: 128 MB Submit: 1468  Solved: 607 [Submit][Status][Discuss] Description 小H最近迷上了一个分隔序列的游戏.在这个游戏里,小H需要将一个长度为n的非负整数序列分割成k+1个非空的子序列.为了得到k+1个子序列,小H需要重复k次以下的步骤: 1.小H首先选择一个长度超过1的序列(一开始小H只有一个长度为n的序列--也就是一

【BZOJ-3675】序列分割 DP + 斜率优化

3675: [Apio2014]序列分割 Time Limit: 40 Sec  Memory Limit: 128 MBSubmit: 1420  Solved: 583[Submit][Status][Discuss] Description 小H最近迷上了一个分隔序列的游戏.在这个游戏里,小H需要将一个长度为n的非负整数序列分割成k+1个非空的子序列.为了得到k+1个子序列,小H需要重复k次以下的步骤: 1.小H首先选择一个长度超过1的序列(一开始小H只有一个长度为n的序列——也就是一开始