[HNOI2008]玩具装箱TOY

20在这里插入图片描述
这题可以说是斜率优化DP的模板题。
首先,我们先推推它的DP式:

f[i]=min{f[j]+(i-(j+1)+c[i]-c[j]-l)2};

这里的c[i]表示的是原来的c[1~i]的和

f[i]表示以i为末尾的箱子的最小费用(前面不一定是全部)

然后我们假设k<j<i,i从j转移比从k转移更优。
那么我们就可以得到一个式子:

f[j]+(i-(j+1)+c[i]-c[j]-l)2<f[k]+(i-(k+1)+c[i]-c[k]-l)2

我们可以将c[i]再表示为c[1~i]+i,这样就简单很多了。
改改DP式:f[i]=min{f[j]+(c[i]-c[j]-l-1)2};
f[j]+(c[i]-c[j]-l-1)2<f[k]+(c[i]-c[k]-l-1)2
我们用换元法,设x=c[j]-l-1,y=c[k]-l-1
f[j]+(c[i]-x)2<f[k]+(c[i]-y)2
f[j]+c[i]2-2 * c[i] * x +x2<f[k]+c[i]2-2 * c[i] * y+y2
f[j]-f[k]+x2-y2<2 * c[i] * (x-y)
f[j]-f[k]+x2-y2<2 * c[i] * (c[j]-c[k])
(f[j]-f[k]+x2-y2)/(c[j]-c[k])<2*c[i]

nice!!!

上标:

#include<cstdio>
#define N 50010
#define db double
#define ll long long
using namespace std;
int n,L,g[N],len=0,l=0;
ll c[N],f[N];db ii;

inline int read()
{
	int x=0; char c=getchar();
	while (c<'0' || c>'9') c=getchar();
	while (c>='0' && c<='9') x=(x<<1)+(x<<3)+(c^48),c=getchar();
	return x;
}

inline ll sqr(ll x) {return x*x;}

db solve(int x,int y)
{
	return (db)(f[x]+sqr(c[x]+L+1)-f[y]-sqr(c[y]+L+1))/(c[x]-c[y]);
}

int main()
{
	freopen("P3195.in","r",stdin);
	freopen("P3195.out","w",stdout);
//	f[i]=min{f[j]+(c[i]-c[j]-L-1)^2};c[i]=c[1~i]+i
	n=read(),L=read();
	for (int i=1;i<=n;i++) c[i]=c[i-1]+read();
	for (int i=1;i<=n;i++) c[i]+=i;
	for (int i=1;i<=n;i++)
	{
		while (l<len && solve(g[l],g[l+1])<2*c[i]) l++;
		f[i]=f[g[l]]+sqr(c[i]-c[g[l]]-L-1);
		while (solve(g[len-1],g[len])>solve(g[len],i) && len) len--;
		g[++len]=i;
	}
	printf("%lld\n",f[n]);
	return 0;
}
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值