[NOIP2019] 划分

题目

题解

首先YY一个最简单的dp

$dp[i][j]=min(dp[j][k]+(sum[i]-sum[j])^2 (sum[i]-sum[j]>=sum[j]-sum[k])$

$dp[i][j]$表示i为终点,j为上一段起点的最小代价。

上述过程其实就是在i的左端寻找满足条件的点,然后一一计算代价并取最小值(注意这些点可能不是连续的)

 

 但其实,我们只要取最右边的满足条件的点来更新就行了,证明见下

 

 我们可以记一个$pre[i]$表示i点上一个区间的终点

这样我们就有了一个$O(n^2)$ dp

对于每个点i,我们从i往前扫,找到第一个满足 $(sum[i]-sum[j]>=sum[j]-sum[pre[j])$ 的点j ,则

$dp[i]=dp[j]+(sum[i]-sum[j])^2$

注意到如果一个点j满足i,那么它肯定满足i+1,

如果不满足i,也可能满足i+1

另外,对于 $sum[i]-sum[j]>=sum[j]-sum[pre[j]$

可以变形成 $sum[i]>=2*sum[j]-sum[pre[j]$

那么可以设一个$d(j)=2*sum[j]-sum[pre[j]$

我们可以用一个单调队列来维护满足条件的点

因为我们只需要最右边满足条件的点,所以每次i递增的时候,如果队头下一位也满足条件,就把队头踢掉(因为下一位肯定比队头更右)

然后用队头更新这一位的dp值

然后把这一位从队尾插入,并把前面d值大于它的点全部删除

这样就可以做到$O(n)$了

代码

这个是88分的,100分的要写高精,但正确性是有的。

#include <iostream>
#include <cstdio>
#define int long long
using namespace std;
#define N 4*(int)1e7+10
int sum[N],pre[N],q[N],dp[N],l=1,r=0;
#define val(a) (2*sum[a]-sum[pre[a]])
signed main()
{
	int n,type;
	cin>>n>>type;
	for(int i=1;i<=n;i++)
	{
		int a;
		scanf("%lld",&a);
		sum[i]=sum[i-1]+a;
	}
	q[++r]=0;
	for(int i=1;i<=n;i++)
	{
		while(l<r&&val(q[l+1])<=sum[i]) l++;
		pre[i]=q[l];
		dp[i]=dp[pre[i]]+(sum[i]-sum[pre[i]])*(sum[i]-sum[pre[i]]);
		while(l<=r&&val(i)<=val(q[r])) r--;
		q[++r]=i;
	}
	cout<<dp[n];
}