[HDU2829]Lawrence

题面描述

传送门

思考

状态转移方程:

F i , p = min ⁡ ( F j , p − 1 + ∑ x = j + 1 i ∑ y = x + 1 i a x ∗ a y ) F_{i,p}=\min(F_{j,p-1}+\sum_{x=j+1}^i\sum_{y=x+1}^ia_x*a_y) Fi,p=min(Fj,p1+x=j+1iy=x+1iaxay)

∑ x = j + 1 i ∑ y = x + 1 i a x ∗ a y = ∑ x = j + 1 i a x ∑ y = x + 1 i a y = ∑ x = j + 1 i a x ( ∑ y = 1 i a y − ∑ y = 1 x a y ) \sum_{x=j+1}^i\sum_{y=x+1}^ia_x*a_y=\sum_{x=j+1}^ia_x\sum_{y=x+1}^ia_y=\sum_{x=j+1}^ia_x(\sum_{y=1}^ia_y-\sum_{y=1}^xa_y) x=j+1iy=x+1iaxay=x=j+1iaxy=x+1iay=x=j+1iax(y=1iayy=1xay)

= ( ∑ x = 1 i a x − ∑ x = 1 j a x ) ( ∑ y = 1 i a y − ∑ y = 1 x a y ) =(\sum_{x=1}^ia_x-\sum_{x=1}^ja_x)(\sum_{y=1}^ia_y-\sum_{y=1}^xa_y) =(x=1iaxx=1jax)(y=1iayy=1xay) = ( ∑ x = 1 i a x − ∑ x = 1 j a x ) ∑ y = 1 i a y − ∑ x = 1 i a x ∑ y = 1 x a y − ∑ x = 1 j a x ∑ y = 1 x a y =(\sum_{x=1}^ia_x-\sum_{x=1}^ja_x)\sum_{y=1}^ia_y-\sum_{x=1}^ia_x\sum_{y=1}^xa_y-\sum_{x=1}^ja_x\sum_{y=1}^xa_y =(x=1iaxx=1jax)y=1iayx=1iaxy=1xayx=1jaxy=1xay

s s s a a a的前缀和,则

( s i − s j ) ∗ s i − ∑ x = 1 i a x ∗ s x + ∑ x = 1 j a x ∗ s x (s_i-s_j)*s_i-\sum_{x=1}^ia_x*s_x+\sum_{x=1}^ja_x*s_x (sisj)six=1iaxsx+x=1jaxsx

s u m sum sum ∑ x = 1 a x ∗ s x \sum_{x=1}a_x*s_x x=1axsx

( s i − s j ) ∗ s i − s u m i + s u m j (s_i-s_j)*s_i-sum_i+sum_j (sisj)sisumi+sumj

之后随便搞搞得了。

F k , p − s i ∗ s k + s u m k ≤ F j , p − s i ∗ s j + s u m j F_{k,p}-s_i*s_k+sum_k\le F_{j,p}-s_i*s_j+sum_j Fk,psisk+sumkFj,psisj+sumj

F k , p − F j , p + s u m k − s u m j ≤ s i ∗ ( s k − s j ) ( s k > s j ) F_{k,p}-F_{j,p}+sum_k-sum_j\le s_i*(s_k-s_j)(s_k>s_j) Fk,pFj,p+sumksumjsi(sksj)(sk>sj)

AC code

#include<cstdio>
#include<algorithm>
#include<cstdlib>
#include<cmath>
#include<cstring>
#define ll long long 
#define gc getchar()
using namespace std;
const int N=1e3+10;
inline void qr(ll &x)
{
    x=0;char c=gc;int f=1;
    while(c<'0'||c>'9'){if(c=='-')f=-1;c=gc;}
    while(c>='0'&&c<='9'){x=x*10+(c^48);c=gc;}
    x*=f;
}
void qw(ll x)
{
    if(x<0)x=-x,putchar('-');
    if(x/10)qw(x/10);
    putchar(x%10+48);
}
ll s[N],sum[N],f[N],g[N],a[N];
int q[N],l,r,n,m;
inline ll calc1(int j,int k)
{
	return g[k]-g[j]+sum[k]-sum[j];
}
inline ll calc2(int j,int k)
{
	return s[k]-s[j];
}
bool pd(int i,int j,int k)
{
	return calc1(j,i)*calc2(k,j)<=calc1(k,j)*calc2(j,i);
}
int main()
{
	while(~scanf("%d%d",&n,&m)&&n&&m)
	{
		memset(s,0,sizeof(s));memset(sum,0,sizeof(sum));m++;
		for(int i=1;i<=n;i++)qr(a[i]),s[i]=s[i-1]+a[i],sum[i]=sum[i-1]+a[i]*s[i];
		for(int i=1;i<=n;i++)f[i]=s[i]*s[i]-sum[i];
		for(int j=2;j<=m;j++)
		{
			memcpy(g,f,n<<3);l=1;r=1;q[1]=0;
			for(int i=1;i<=n;i++)
			{
				while(l<r&&calc1(q[l],q[l+1])<=s[i]*calc2(q[l],q[l+1]))++l;
				while(l<r&&pd(i,q[r],q[r-1]))--r;
				q[++r]=i;f[i]=g[q[l]]+(s[i]-s[q[l]])*s[i]-sum[i]+sum[q[l]];
			}
		}
		qw(f[n]);puts("");
	}
	return 0;
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值