【BZOJ3437】小P的牧场(动态规划,斜率优化)

题面

BZOJ

题解

考虑暴力\(dp\),设\(f[i]\)表示强制在\(i\)处建立控制站的并控制\([1..i]\)的最小代价。

很显然,枚举上一个控制站的位置\(j\)

\(f[i]=min(f[j]+Calc(i,j)+a[i])\),其中\(Calc(i,j)\)表示\(i,j\)之间被\(i\)控制的位置产生的贡献。

这个可以用前缀和优化做到\(O(1)\)计算\(Calc\)

预处理\(s1[i]=\sum b[i],s2[i]=\sum (n-i+1)*b[i]\)

那么\(Calc(i,j)=s2[i]-s2[j]-(s1[i]-s1[j])*(n-i+1)\)

考虑两个位置\(j,k\),满足\(k\lt j\),并且\(k\)的转移劣于\(j\)

那么

\(f[k]+Calc(i,k)\gt f[j]+Calc(i,j)\)

拆开之后是:

\(f[k]-s2[k]+s1[k]*(n-i+1)\gt f[j]-s2[j]+s1[j]*(n-i+1)\)

令\(g[i]=f[i]-s2[i]+s1[i]*(n+1)\)

将所有项按照是否与\(i\)相关分类,可以得到

\((g[k]-g[j])\gt (s1[k]-s1[j])*i\)

因为\(k<j\),所以\(s1[k]<s1[j]\),除过去要变号

\[i>\frac{g[k]-g[j]}{s1[k]-s1[j]}
\]

妥妥的斜率优化,因为\(i\)单增,可以单调队列解决。

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
using namespace std;
#define ll long long
#define RG register
#define MAX 1000100
inline int read()
{
RG int x=0,t=1;RG char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=-1,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return x*t;
}
int n,a[MAX],b[MAX];
int Q[MAX],h,t;
ll f[MAX],s1[MAX],s2[MAX];
ll calc(int i,int j){return f[j]+s2[i]-s2[j]-(s1[i]-s1[j])*(n-i+1)+a[i];}
double Slope(int j,int k)
{
double gj=f[j]-s2[j]+1.0*s1[j]*(n+1);
double gk=f[k]-s2[k]+1.0*s1[k]*(n+1);
return 1.0*(gj-gk)/(s1[j]-s1[k]);
}
int main()
{
n=read();
for(int i=1;i<=n;++i)a[i]=read();
for(int i=1;i<=n;++i)b[i]=read();
for(int i=1;i<=n;++i)s1[i]=s1[i-1]+b[i];
for(int i=1;i<=n;++i)s2[i]=s2[i-1]+1ll*(n-i+1)*b[i];
Q[h=t=1]=0;
for(int i=1;i<=n;++i)
{
while(h<t&&Slope(Q[h],Q[h+1])<=i)++h;
int j=Q[h];f[i]=calc(i,j);
while(h<t&&Slope(Q[t],Q[t-1])>=Slope(Q[t-1],i))--t;
Q[++t]=i;
}
printf("%lld\n",f[n]);
}

最新文章

  1. 把int*传值给char*,打印出错误的数字
  2. 一步步开发自己的博客 .NET版(5、Lucenne.Net 和 必应站内搜索)
  3. Python中字符串操作
  4. 分享php中四种webservice实现的简单架构方法及实例
  5. 安卓开发_浅谈Service
  6. rcp(插件开发) 如何查找自己定义的扩展点
  7. 关于js向jsp中传输中文乱码问题
  8. mysql 数据库优化要点
  9. Swing-JTable的渲染器与编辑器使用demo
  10. MySQL字段操作与数据处理
  11. (92)Wangdao.com_第二十五天_线程机制_H5 Web Workers 分线程任务_事件 Event
  12. lnmp----------lnmp集成环境使用lnmp安装包安装lnmp集成环境的步骤
  13. docker的完整解决方案2
  14. 卸载或重新安装Redis集群
  15. mfc CString,string,char* 之间的转换
  16. Window环境下RabbitMQ 添加用户、设置角色和权限
  17. java图形用户界面之列表框
  18. 000. 规范类的设计(ing)
  19. SpringMVC框架 注解 (转)
  20. Lucene.Net 3.0.3如何从TokenStream中获取token对象

热门文章

  1. web存储机制(localStorage和sessionStorage)
  2. TW实习日记:第五天
  3. Keepalived两节点出现双VIP的情况
  4. Apriori 获取关联规则实现
  5. POWERDESIGNER生成的代码有引号
  6. Sublime Text 3高效实用快捷键
  7. ERROR [IM002] [Microsoft][ODBC 驱动程序管理器] 未发现数据源名称并且未指定默认驱动程序
  8. 面对30页左右的运放数据手册datasheet,你需要知道如何看懂
  9. C++:this指针的简单理解
  10. 【IdentityServer4文档】- 启动和概览