【bzoj3437】小P的牧场 斜率优化dp

题目描述

背景

小P是个特么喜欢玩MC的孩纸。。。

描述

小P在MC里有n个牧场,自西向东呈一字形排列(自西向东用1…n编号),于是他就烦恼了:为了控制这n个牧场,他需要在某些牧场上面建立控制站,每个牧场上只能建立一个控制站,每个控制站控制的牧场是它所在的牧场一直到它西边第一个控制站的所有牧场(它西边第一个控制站所在的牧场不被控制)(如果它西边不存在控制站,那么它控制西边所有的牧场),每个牧场被控制都需要一定的花费(毕竟在控制站到牧场间修建道路是需要资源的嘛~),而且该花费等于它到控制它的控制站之间的牧场数目(不包括自身,但包括控制站所在牧场)乘上该牧场的放养量,在第i个牧场建立控制站的花费是ai,每个牧场i的放养量是bi,理所当然,小P需要总花费最小,但是小P的智商有点不够用了,所以这个最小总花费就由你来算出啦。

输入

第一行一个整数 n 表示牧场数目

第二行包括n个整数,第i个整数表示ai

第三行包括n个整数,第i个整数表示bi

输出

只有一行,包括一个整数,表示最小花费

样例输入

4
2 4 2 4
3 1 4 2

样例输出

9


题解

斜率优化dp

设f[i]为i建立控制站时前i个的最小代价。

那么有f[i]=f[j]+∑((i-k)*b[k])+a[i] (j+1≤k≤i)

             =f[j]+∑(i*b[k])-∑(k*b[k])+a[i] (j+1≤k≤i)

             =f[j]+i*(sum[i]-sum[j])-(t[i]-t[j])+a[i]

其中sum[i]为b[i]的前缀和,t[i]为b[i]*i的前缀和。

整理一下即为f[j]+t[j]=i*sum[j]+f[i]-i*sum[i]+t[i]-a[i]。

这样状态转移方程就让我们转化成y=kx+b的形式,并且要求f[i]的最小值,就是求这里b的最小值。

于是维护一个下凸包即可。

#include <cstdio>
#define y(i) (f[i] + t[i])
#define x(i) sum[i]
long long f[1000010] , a[1000010] , b[1000010] , sum[1000010] , t[1000010];
int q[1000010] , l , r;
int main()
{
	int n , i;
	scanf("%d" , &n);
	for(i = 1 ; i <= n ; i ++ )
		scanf("%lld" , &a[i]);
	for(i = 1 ; i <= n ; i ++ )
		scanf("%lld" , &b[i]) , sum[i] = sum[i - 1] + b[i] , t[i] = t[i - 1] + b[i] * i;
	for(i = 1 ; i <= n ; i ++ )
	{
		while(l < r && y(q[l + 1]) - y(q[l]) < (x(q[l + 1]) - x(q[l])) * i) l ++ ;
		f[i] = y(q[l]) - i * x(q[l]) + i * sum[i] - t[i] + a[i];
		while(l < r && (y(i) - y(q[r])) * (x(q[r]) - x(q[r - 1])) < (x(i) - x(q[r])) * (y(q[r]) - y(q[r - 1]))) r -- ;
		q[++r] = i;
	}
	printf("%lld\n" , f[n]);
	return 0;
}

 

posted @ 2017-02-23 18:13  GXZlegend  阅读(278)  评论(0编辑  收藏  举报