bzoj4518

斜率优化。

把m^2乘进式子里,优化式子。

注意初始化。

#include<cstdio>
#include<cstdlib>
#include<cstring>
long long a[3005];
long long f[3005][3005];
long long s[3005];
long long q[3005];
int t; 
double Y(int j)
{
    return (double)(f[t-1][j]+s[j]*s[j]);
}
double X(int j)
{
    return (double)s[j];    
}
double slop(int j1,int j2)
{
 return (Y(j2)-Y(j1))/(X(j2)-X(j1));    
}
int main()
{
 int n,m;
 scanf("%d%d",&n,&m);
 for (int i=1;i<=n;i++)
 {
 scanf("%lld",&a[i]);
 s[i]=s[i-1]+a[i];
 }
 memset(f,63,sizeof(f));
 f[0][0]=0;
 for (t=1;t<=m;t++)
 {
  int l=1,r=1;
  q[1]=0;
  for (int i=1;i<=n;i++)
  {
   while(l<r&&slop(q[l],q[l+1])<2.0*s[i])l++;
   int k=q[l];
   f[t][i]=f[t-1][k]+(s[i]-s[k])*(s[i]-s[k]);
   while(l<r&&slop(q[r],q[r-1])>slop(q[r],i))r--;
   q[++r]=i;
  }
 }
 long long ans=f[m][n]*m-s[n]*s[n];
 printf("%lld",ans);
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值