本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作。

本文作者:ljh2000
作者博客:http://www.cnblogs.com/ljh2000-jump/
转载请注明出处,侵权必究,保留最终解释权!

Description

  P教授要去看奥运,但是他舍不下他的玩具,于是他决定把所有的玩具运到北京。他使用自己的压缩器进行压
缩,其可以将任意物品变成一堆,再放到一种特殊的一维容器中。P教授有编号为1...N的N件玩具,第i件玩具经过
压缩后变成一维长度为Ci.为了方便整理,P教授要求在一个一维容器中的玩具编号是连续的。同时如果一个一维容
器中有多个玩具,那么两件玩具之间要加入一个单位长度的填充物,形式地说如果将第i件玩具到第j个玩具放到一
个容器中,那么容器的长度将为 x=j-i+Sigma(Ck) i<=K<=j 制作容器的费用与容器的长度有关,根据教授研究,
如果容器长度为x,其制作费用为(X-L)^2.其中L是一个常量。P教授不关心容器的数目,他可以制作出任意长度的容
器,甚至超过L。但他希望费用最小.

Input

  第一行输入两个整数N,L.接下来N行输入Ci.1<=N<=50000,1<=L,Ci<=10^7

Output

  输出最小费用

Sample Input

5 4
3
4
2
1
4

Sample Output

1

正解:决策单调性+DP+单调队列

解题报告:

  许久之前的一道考试题目了,题解直接用学长的了:

  

  首先贪心明显是错的(虽然可以过样例),想找反例的话随便看一个测试数据吧。
  由于同一容器中玩具的编号必须是连续的,因此满足无后效性,可用动态规划解决。一个比较容易想到O(N^2 )的算法是,用F i 表示装前i个玩具的最少费用,S i 表示前i个玩具的总长度,则状态
  转移方程为:
  F i = min {F j + (S i − S j + i − (j + 1) − L) 2 }0≤j<i
  但由于N 最大可达到50000,所以期望得分为10∼20分。
  然而,真正颇为牛B的东西来了。
  令S i ′ = S i + i, L ′ = L + 1,让我们重新观察原方程:
  F i = min {F j + (S i ′ − S j ′ − L ′ ) 2 } 0≤j<i
  令V j = F j + (S i ′ − S j ′ − L ′ ) 2 ,那么在x < y时,V x ≥ V y 可变形为:
  F x − F y ≥ (S i ′ − S y ′ − L ′ ) 2 − (S i ′ − S x ′ − L ′ ) 2
  ⇒ F x − F y ≥ (2S i ′ − S x ′ − S y ′ − 2L ′ )(S x ′ − S y ′ )
  ⇒ F x − F y ≥ (2(S i ′ − L ′ ) − (S x ′ + S y ′ ))(S x ′ − S y ′ )
  ⇒ F x − F y ≥ 2(S i ′ − L ′ )(S x ′ − S y ′ ) − (S x ′ + S y ′ )(S x ′ − S y ′ )
  ⇒ F x − F y + S x ′ − S y ′ ≥ 2(S i ′ − L ′ )(S x ′ − S y ′ )
  这样的话,如果对于当前的i,有x < y, V x ≥ V y ,那么对于以后更大的i,由于2(S i ′ − L ′ )的单调性,V x ≥ V y
  将会恒成立,也就是说x不可能再成为最优决策,我们将永远不会再用到它。

  因为决策i可能比原队列中的某些决策更优,如果不将这些肯定没用的决策删除,队列前面的非最优决策可能受到它们当中的某个决策“保护”而没有被删除,其实这样也就是保证函数G的值在Q中单调递增。
  由于每个元素最多入队一次,出队一次,所以维护队列的时间复杂度为O(N ),状态转移为O(1),计算F 值为O(N ),所以总时间复杂度为O(N ) + O(1) × O(N ) = O(N )。

 //It is made by jump~
#include <iostream>
#include <cstdlib>
#include <cstring>
#include <cstdio>
#include <cmath>
#include <algorithm>
#include <ctime>
#include <vector>
#include <queue>
#include <map>
#include <set>
using namespace std;
typedef long long LL;
const LL inf = (1LL<<);
const int MAXN = ;
int n,a[MAXN],dui[MAXN],head,tail;
LL L,f[MAXN],sum[MAXN],S[MAXN]; inline int getint()
{
int w=,q=; char c=getchar();
while((c<'' || c>'') && c!='-') c=getchar(); if(c=='-') q=,c=getchar();
while (c>='' && c<='') w=w*+c-'', c=getchar(); return q ? -w : w;
} inline bool check(int x,int y,int i){
x=dui[x]; y=dui[y];
LL ji=f[x]-f[y]+S[x]*S[x]-S[y]*S[y];
LL comp=*(S[i]-L); comp*=S[x]-S[y];
if(ji>=comp) return true; return false;
} inline bool check2(int x,int y,int i){
x=dui[x]; y=dui[y];
LL ji1=f[x]-f[y]+S[x]*S[x]-S[y]*S[y]; ji1/=(S[x]-S[y]);
LL ji2=f[y]-f[i]+S[y]*S[y]-S[i]*S[i]; ji2/=(S[y]-S[i]);
if(ji1>=ji2) return true; return false;
} inline void work(){
n=getint(); L=getint(); for(int i=;i<=n;i++) a[i]=getint(),sum[i]=sum[i-]+a[i],S[i]=sum[i]+i;
head=tail=; dui[head]=; L++;
for(int i=;i<=n;i++) {
while(head<tail && check(head,head+,i)) head++;
f[i]=f[dui[head]]+(S[i]-S[dui[head]]-L)*(S[i]-S[dui[head]]-L);
while(head<tail && check2(tail-,tail,i)) tail--;
dui[++tail]=i;
}
printf("%lld",f[n]);
} int main()
{
work();
return ;
}

BZOJ1010 [HNOI2008]玩具装箱toy的更多相关文章

  1. bzoj1010[HNOI2008]玩具装箱toy 斜率优化dp

    1010: [HNOI2008]玩具装箱toy Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 11893  Solved: 5061[Submit][S ...

  2. bzoj1010: [HNOI2008]玩具装箱toy(DP+斜率优化)

    1010: [HNOI2008]玩具装箱toy 题目:传送门 题解: 很明显的一题动态规划... f[i]表示1~i的最小花费 那么方程也是显而易见的:f[i]=min(f[j]+(sum[i]-su ...

  3. [BZOJ1010] [HNOI2008] 玩具装箱toy (斜率优化)

    Description P教授要去看奥运,但是他舍不下他的玩具,于是他决定把所有的玩具运到北京.他使用自己的压缩器进行压缩,其可以将任意物品变成一堆,再放到一种特殊的一维容器中.P教授有编号为1... ...

  4. [bzoj1010](HNOI2008)玩具装箱toy(动态规划+斜率优化+单调队列)

    Description P教授要去看奥运,但是他舍不下他的玩具,于是他决定把所有 的玩具运到北京.他使用自己的压缩器进行压缩,其可以将任意物品变成一堆,再放到一种特殊的一维容器中.P教授有编号为1.. ...

  5. [BZOJ1010][HNOI2008]玩具装箱toy 解题报告

    Description P教授要去看奥运,但是他舍不下他的玩具,于是他决定把所有的玩具运到北京.他使用自己的压缩器进行压缩,其可以将任意物品变成一堆,再放到一种特殊的一维容器中.P教授有编号为1... ...

  6. BZOJ1010 [HNOI2008]玩具装箱toy 动态规划 斜率优化

    原文链接http://www.cnblogs.com/zhouzhendong/p/8687797.html 题目传送门 - BZOJ1010 题意 一个数列$C$,然后把这个数列划分成若干段. 对于 ...

  7. 2018.09.05 bzoj1010: [HNOI2008]玩具装箱toy(斜率优化dp)

    传送门 一道经典的斜率优化dp. 推式子ing... 令f[i]表示装前i个玩具的最优代价. 然后用老套路. 我们只考虑把第j+1" role="presentation" ...

  8. 题解【bzoj1010 [HNOI2008]玩具装箱TOY】

    斜率优化动态规划可以用来解决这道题.同时这也是一道经典的斜率优化基础题. 分析:明显是动态规划.令\(dp[i]\)为前\(i\)个装箱的最小花费. 转移方程如下: \[dp[i]=\min\limi ...

  9. 【斜率优化】BZOJ1010 [HNOI2008]玩具装箱toy

    [题目大意] P教授有编号为1...N的N件玩具,第i件玩具长度为Ci.为了方便整理,P教授要求在一个一维容器中的玩具编号是连续的.如果将第i件玩具到第j个玩具放到一 个容器中,那么容器的长度将为 x ...

随机推荐

  1. 044医疗项目-模块四:采购单模块—采购单保存(Dao,Service,Action三层)

    我们上上一篇文章(042医疗项目-模块四:采购单模块-采购单明细添加查询,并且把数据添加到数据库中)做的工作是把数据插入到了数据库,我们这篇文章做的是042医疗项目-模块四:采购单模块-采购单明细添加 ...

  2. AngularJS中的控制器和作用域

    欢迎大家指导与讨论 : ) 一. 作用域的事件传播 一 . 1 修改的传播   关于作用域最重要的一点是修改会通过事件传播下去,自动更新所以依赖的数据值,即使是通过行为产生的.简而言之,就是即时您只修 ...

  3. CodeDom

    细说CodeDom 在上一篇文章中,老周厚着脸皮给大伙介绍了代码文档的基本结构,以及一些代码对象与CodeDom类型的对应关系. 在评论中老周看到有朋友提到了 Emit,那老周就顺便提一下.严格上说, ...

  4. 学习C++.Primer.Plus 5 循环和关系表达式

    C++将赋值表达式的值定义为左侧成员的值 赋值操作符是自右向左结合的 cout.setf(ios:: boolalpha);//调用设置标记,命令cout输出true或false,而非1或0. 任何表 ...

  5. KMS10流氓软件

    win10想激活,结果中了流氓软件的当... (关键是win10家庭单语言版居然还激活不了....白吃亏了) 把我的chrome 和 firefox 主页改成hao.qquu8.com,该网址重定向到 ...

  6. 快速开发之代码生成器(asp.net mvc4 + easyui + knockoutjs)

    一.前言 作为一个码农这么多年,一直在想怎么提高我们的编码效率,关于如何提高编码效率,我自己的几点体会 1.清晰的项目结构,要编写代码的地方集中 2.实现相同功能的代码量少并且清晰易懂 3.重复或有规 ...

  7. 用canvas画简单的“我的世界”人物头像

    前言:花了4天半终于看完了<Head First HTML5>,这本书的学习给我最大的感受就是,自己知识的浅薄,还有非常多非常棒的技术在等着我呢.[熊本表情]扶朕起来,朕还能学! H5新增 ...

  8. 细数Javascript技术栈中的四种依赖注入

    作为面向对象编程中实现控制反转(Inversion of Control,下文称IoC)最常见的技术手段之一,依赖注入(Dependency Injection,下文称DI)可谓在OOP编程中大行其道 ...

  9. 我做PHP,但是我要批判下整天唱衰.NET的淫

    笔者每天都能看到月经贴-".NET已死"!!! 笔者之前一直在CSDN上面写博客,泡论坛,但是有约莫一年来着了发现CSDN上面的博客都没啥更新,CSDN首页推荐的一些文章也没啥新意 ...

  10. 对react的几点质疑

    现在react.js如火如荼,非常火爆,昨天抽了一天来看了下这项技术.可能就看了一天,研究的不深入,但是我在看的过程中发现来了很多疑惑,这里拿出来和那家分享讨论以此共勉. 在我接触的前端以后,让我感觉 ...