背景:
-
有一天,你打开了这道题
-
你推出了式子
-
你写完了斜率优化
-
一交, Unaccepted 100
-
于是,你心态炸了
做法:
首先,先把 dp 式子推出来:
设 {f_i}fi 为 在 { i }i 处建立基地的情况下,从 {1}1 处理到 { i }i 时的最小总代价。
那么有 { f_i = \min \limits_{ 0 \le j < i } \left \{ { { f_j + x_i \times \sum \limits_{ l=j+1 } ^ {i}{p_l} } - \sum \limits_{ l=j+1 } ^ { i } { x_l \times p_l } } \right \} + c_i }fi=0≤j<imin{fj+xi×l=j+1∑ipl−l=j+1∑ixl×pl}+ci 。
然后,考虑前缀和优化。记 {sump_i}sumpi 为 {p_i}pi 的前缀和, {sum_i}sumi 为 { p_i \times x_i}pi×xi 的前缀和。
那么 {f_i = \min \limits_{0 \le j < i} \left \{ f_j + x_i \times (sump_i - sump_j) - ( sum_i - sum_j ) \right \} + c_i}fi=0≤j<imin{fj+xi×(sumpi−sumpj)−(sumi−sumj)}+ci 。
展开,移项,考虑 {f_i}fi 与 {f_j}fj 的关系 ,可得 { f_j + sum_j = f_i - x_i \times sump_j - c_i }fj+sumj=fi−xi×sumpj−ci 。
将 ( sump_j , f_j + sum_j)(sumpj,fj+sumj) 看作平面上的点。
然后就可以快乐地斜率优化了。
然后,你就知道背景是什么意思了。
这里膜拜出 Hack 数据的神,脑洞是真的大 。
我们还要考虑一种情况:若最后几处没有货物的话,不一定要在 {n}n 处建立仓库。
我们应该从 {n}n 往上找,直到找到 {p_i}pi 非零处,记为 {mem}mem 。
那么正确答案就是 { \min \limits_{ mem \le i \le n } { f_i} }mem≤i≤nminfi 。
献上代码
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
#define _for(i,a,b) for(register int (i)=(a);(i)<=(b);(i)++)
#define For(i,a,b) for(register int (i)=(a);(i)>=(b);(i)--)
#define INF 0x7fffffff
#define il inline
#define rg register
#define int ll
const int N=1e6+5;
int n;
ll sp[N],s[N],f[N];
int x[N],p[N],c[N],q[N],l,r;
#define Y(o) (f[o]+s[o])
#define X(o) (sp[o])
il double slope(rg int a,rg int b){
if(X(b)==X(a)) return 1e18;
return 1.0*(Y(b)-Y(a))/(X(b)-X(a));
}
signed main(){
scanf("%d",&n);
_for(i,1,n) scanf("%lld%lld%lld",x+i,p+i,c+i), sp[i]=sp[i-1]+p[i], s[i]=s[i-1]+x[i]*p[i];
q[l=r=1]=0;
_for(i,1,n){
while(l<r and slope(q[l],q[l+1])<=(double)1.0*x[i]) ++l;
rg int j(q[l]);
f[i]=f[j]+sp[i]*x[i]-sp[j]*x[i]-s[i]+s[j]+1LL*c[i];
while(l<r and slope(q[r-1],q[r])>=slope(q[r],i)) --r;
q[++r]=i;
}
rg int mem(n);
while(!p[mem] and mem) --mem;
if(!mem) return puts("0"), 0;
rg ll ans(1e18);
_for(i,mem,n) ans=min(ans,f[i]);
return printf("%lld\n",ans),0;
}