洛谷P3195 玩具装箱TOY
题目大意:
有n个数,要将他们分成若干段,每一段的cost定义为: cost=r-l+ΣCk (k∈[r,l]) 该段的最终花费是:(cost-L)^2; 给出L,n,C(1~n),总共的最小花费。
分析:
dp方程极容易想出来: f[i]=max(f[j]+(sum[i]-sum[j]+i-j-1-L)^2) 其中sum[i]表示c(1~i)的和。因为取的这一段数从j+1开始,所以i-j-1(题目中i-j并不是区间长度!没有再加1)
O(n^2)直接挂掉。
因为状态O(n)已经非常不错,无法再优化了。所以考虑能不能优化转移的O(n)。
将表达式展开:
f[i]=f[j]+(sum[i]-sum[j]+i-j-1-L)^2
令a[k]=sum[k]+k;x[k]=a[k]+1+ f[i]=f[j]+(a[i]-x[j])^2
f[i]=f[j]+x[j]^2-2a[i]x[j]+a[i]^2
对于给定的i,a[i]^2是一个定值,所以当做常数先不用管,但是记得最后加上!
令y[k]=f[k]+x[k]^2; f[i]=y[j]-2a[i]x[j]
移项: y[j]=2a[i]x[j]+f[i]
对于给定的i,我们需要循环所有的j。 对于一个j,我们已经知道了x[j],y[j]。 将它看作一个点,所有的j构成一个点集,横坐标x[j],纵坐标y[j]
而对于给定的i,2a[i]是一个定值,看做斜率,而目标f[i]则是截距,所以要在j的点集之中找到一个点使得这条直线截距最小,本质是将y=2a[i]x的直线平移。
可以发现这些最优解的点必定在边界上。并且发现,斜率是单增的,而x一定也是单增的。所以可以维护一个左上下凸壳。
因为斜率单增,所以若一个点此时不是最优解,那么以后一定也不是最优解,可以直接从头pass掉;每次新增一个点,都要保证这是一个凸包,通过斜率要来从尾pass掉。
所以可以用单调队列维护!单调,在这里是指队列中的元素,每相邻两个元素所代表的点连成的线,其斜率是单调递增的。因为斜率都是正数,所以也就是画出来是一个左上凸包。
注意,队列中至少要有一个0号元素不能出队!这里第一个空位置是有意义的。否则就不存在线和斜率了。所以手写队列时,开始hd=tl=0,while判断中hd<tl 保证一定至少有一个元素,并且这个元素不会被其他元素替代。详见代码。
具体步骤:
1.前缀和预处理
2.循环i,先删除队首,再更新答案,再更新队列。
3.输出f[n] GAME OVER
代码:
#include<bits/stdc++.h>
#define ll long long
using namespace std;
const int N=+;
int n,L;
long long f[N];
long long sum[N];
int q[N],hd,tl;
long long a(int i)
{
return sum[i]+i;
}
long long x(int i)
{
return sum[i]+i+L+;
}
long long y(int i)
{
return x(i)*x(i)+f[i];
}
double slope(int a,int b)
{
return ((double)y(b)-y(a))/((double)x(b)-x(a));
}//一堆函数
int t;
int main()
{
scanf("%d%d",&n,&L);
for(int i=;i<=n;i++)
{
scanf("%d",&t);
sum[i]=sum[i-]+t;
}
hd=,tl=;
for(int i=;i<=n;i++)
{
while(hd<tl&&slope(q[hd],q[hd+])<*a(i)) hd++;//删除
f[i]=y(q[hd])-*a(i)*x(q[hd])+a(i)*a(i);
while(hd<tl&&slope(q[tl-],q[tl])>slope(q[tl-],i)) tl--;
q[++tl]=i;//更新
}
printf("%lld",f[n]);
return ;
}
总结:
1.一个题能用斜率优化,必然能出现y=kx+b 线性形式,其中k是定值,x,y构成点集,b是目标值。
2.一个凸包能用单调队列优化的条件应该满足:
(1)查询的斜率单调 (2)插入的点横坐标有单调性 (否则要用平衡树、set)
洛谷P3195 玩具装箱TOY的更多相关文章
- 洛谷P3195 玩具装箱
P3195 [HNOI2008]玩具装箱TOY 第一道斜率优化题. 首先一个基本的状态转移方程是 要使f[i]最小,即b最小. 对于每个j,可以表示为一个点. 然后我们取固定斜率时截距最小的即可,高中 ...
- 洛谷P3195||bzoj1010 [HNOI2008]玩具装箱TOY
洛谷P3195 bzoj1010 设s数组为C的前缀和 首先$ans_i=min_{j<i}\{ans_j+(i-j-1+s_i-s_j-L)^2\}$ (斜率优化dp)参考(复读)https: ...
- 斜率优化dp学习笔记 洛谷P3915[HNOI2008]玩具装箱toy
本文为原创??? 作者写这篇文章的时候刚刚初一毕业…… 如有错误请各位大佬指正 从例题入手 洛谷P3915[HNOI2008]玩具装箱toy Step0:读题 Q:暴力? 如果您学习过dp 不难推出d ...
- P3195 [HNOI2008]玩具装箱TOY(斜率优化dp)
P3195 [HNOI2008]玩具装箱TOY 设前缀和为$s[i]$ 那么显然可以得出方程 $f[i]=f[j]+(s[i]-s[j]+i-j-L-1)^{2}$ 换下顺序 $f[i]=f[j]+( ...
- [luogu P3195] [HNOI2008]玩具装箱TOY
[luogu P3195] [HNOI2008]玩具装箱TOY 题目描述 P教授要去看奥运,但是他舍不下他的玩具,于是他决定把所有的玩具运到北京.他使用自己的压缩器进行压缩,其可以将任意物品变成一堆, ...
- BZOJ 1010: [HNOI2008]玩具装箱toy [DP 斜率优化]
1010: [HNOI2008]玩具装箱toy Time Limit: 1 Sec Memory Limit: 162 MBSubmit: 9812 Solved: 3978[Submit][St ...
- 【BZOJ-1010】玩具装箱toy DP + 斜率优化
1010: [HNOI2008]玩具装箱toy Time Limit: 1 Sec Memory Limit: 162 MBSubmit: 8432 Solved: 3338[Submit][St ...
- BZOJ 1010: [HNOI2008]玩具装箱toy 斜率优化DP
1010: [HNOI2008]玩具装箱toy Description P教授要去看奥运,但是他舍不下他的玩具,于是他决定把所有的玩具运到北京.他使用自己的压缩器进行压缩,其可以将任意物品变成一堆,再 ...
- 【BZOJ】【1010】【HNOI2008】玩具装箱Toy
DP/斜率优化 根据题目描述很容易列出动规方程:$$ f[i]=min\{ f[j]+(s[i]-s[j]+i-j-1-L)^2 \}$$ 其中 $$s[i]=\sum_{k=1}^{i} c[k] ...
随机推荐
- PHP实现验证码制作
captcha.php(PHP产生验证码并储存Session): <?php //开启Session session_start(); //绘制底图 $image = imagecreatetr ...
- awk技巧(如取某一行数据中的倒数第N列等)
使用awk取某一行数据中的倒数第N列:$(NF-(n-1))比如取/etc/passwd文件中的第2列.倒数第1.倒数第2.倒数第4列(以冒号为分隔符) [root@ipsan-node06 ~]# ...
- 分布式监控系统Zabbix-3.0.3--短信报警设置
前面已分别介绍了zabbix的邮件.微信报警设置,这些都是手机在有网络时才能收到报警信息,那如果手机没有网的情况下怎么办,这就需要考虑使用短信接口报警了.当服务出现故障达到预警级别是通过发送短信的形式 ...
- javaScript常用API合集
节点 1.1 节点属性 Node.nodeName //返回节点名称,只读 Node.nodeType //返回节点类型的常数值,只读 Node.nodeValue //返回Text或Com ...
- Linux内核及分析 第三周 Linux内核的启动过程
实验过程: 打开shell终端,执行以下命令: cd LinuxKernel/ qemu -kernel linux-3.18.6/arch/x86/boot/bzImage-initrd rootf ...
- 《Linux内核分析》第六周学习笔记
<Linux内核分析>第六周学习笔记 进程的描述和创建 郭垚 原创作品转载请注明出处 <Linux内核分析>MOOC课程http://mooc.study.163.com/co ...
- 《Linux内核设计与实现》读书笔记六
第4章 进程调度35 调度程序负责决定将哪个进程投入运行,何时运行以及运行多长时间,进程调度程序可看做在可运行态进程之间分配有限的处理器时间资源的内核子系统.只有通过调度程序的合理调度,系统资源才能最 ...
- svn 使用教程
一.什么是SVN SVN是Subversion的简称,是一个开放源代码的版本控制系统,相较于RCS.CVS,它采用了分支管理系统,它的设计目标就是取代CVS. 二.SVN的下载安装 下载地址:http ...
- 『编程题全队』Beta 阶段冲刺博客一
1.提供当天站立式会议照片一张 2.每个人的工作 (有work item 的ID) (1) 昨天已完成的工作 孙志威: 1.讨论并制定了Beta阶段的计划 孙慧君: 1.Beta阶段任务的认领 黄华林 ...
- Python入门:如何使用第三方库?
这是关于Python的第13篇文章,也是关于<编程小白的第1本Python入门书>内容的最后一篇,主要介绍下如何使用第三方库. 1. 第三方库 Python相当于一个手机,第三方库相当于手 ...