题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3507

Zero has an old printer that doesn't work well sometimes. As it is antique, he still like to use it to print articles. But it is too old to work for a long time and it will certainly wear and tear, so Zero use a cost to evaluate this degree. 
One day Zero want to print an article which has N words, and each word i has a cost Ci to be printed. Also, Zero know that print k words in one line will cost

M is a const number. 
Now Zero want to know the minimum cost in order to arrange the article perfectly. 

Input

There are many test cases. For each test case, There are two numbers N and M in the first line (0 ≤ n ≤ 500000, 0 ≤ M ≤ 1000). Then, there are N numbers in the next 2 to N + 1 lines. Input are terminated by EOF.

Output

A single number, meaning the mininum cost to print the article.

Sample Input

5 5
5
9
5
7
5

Sample Output

230

题意:

假设一篇文章有N个词,每个词有c[i]代表花费;

现在打印文章,打印一行需要花费 M + ( ∑c[i] )2,求怎么打印整篇文章花费最少,为多少。

题解:

假设dp[i]代表输出前i个词的最小费用,那么就有状态转移方程:

sum[i]代表前i个词的c[i]之和;

状态转移方程的意思也很明显,就是枚举选出第i个单词所在的那一行上打印多少个单词最优;

那么如果直接求解的话,O(n2)的时间复杂度对于n=500000的规模是超时的,所以就要使用斜率优化:

那么当我们计算dp[i]时,假设有 1 <= k < j < i,如果有:

就可以说j点比k点更优。

对上式进行变形可得:

我们设

那么就有:(在计算dp[i]时)j点比k点优  ↔  g(k,j) ≤ sum[i];……………………(1)

              j点比k点差  ↔  g(k,j) > sum[i];……………………(2)

(这就像,g(k,j)代表k点与j点的连线斜率,这个斜率小于某个阈值就j更优,否则就k更优)

就不难得到:

①在计算dp[i]时,若有1≤a<b<c<i,只要满足g(a,b) ≥ g(b,c),则b点必然被淘汰.  (注意此处的g(a,b) ≥ g(b,c)是必须是要取到等号的)

证明:若g(b,c) ≤ sum[i],由(1)得c点优于b点,选择c点;

   若g(b,c) > sum[i],则g(a,b) ≥ g(b,c) > sum[i],由(2)得a点比b点更优,选择a点;

   最终,怎么样都选不到b点,故b点必然淘汰。

②若在计算dp[i]时,k点被淘汰,则计算dp[i+1]时,k点必然也被淘汰.

证明:当k点被淘汰,即必然有一点j比k更优,则g(k,j) ≤ sum[i],由于c[i] > 0,则g(k,j) ≤ sum[i] < sum[i+1],即计算dp[i+1]时,j依然比k更优,k依然淘汰。

因此我们可以考虑维护一个斜率的队列来优化整个DP过程:

1、求解dp[i]:

  假设队列中的头部依次存在元素a,b,且g(a,b) ≤ sum[i],则b点比a点优,于是a淘汰出队;

  重复上述操作,直到队列元素个数小于2或者g(a,b) > sum[i]。

  最后dp[i]=getDP(Q[head])。

2、入队点i:

  假设队列中的尾部依次存在元素a,b,那么当点c要入队的时候,如果g(a,b) > g(b,c),那么就将b点删除;

  重复上述操作,直到队列元素个数小于2,或者找到x,y能满足g(x,y) ≤ g(y,c),将d点加入在该位置中。

AC代码:

#include<bits/stdc++.h>
using namespace std;
const int maxn=; int n,m,sum[maxn];
int dp[maxn];
int q[maxn],head,tail; //数组模拟队列 int getDP(int i,int j){return dp[j]+(sum[i]-sum[j])*(sum[i]-sum[j])+m;} int gUp(int k,int j){return (dp[j]+sum[j]*sum[j])-(dp[k]+sum[k]*sum[k]);} //g(k,j)的分子部分
int gDown(int k,int j){return *(sum[j]-sum[k]);} //g(k,j)的分母部分 int main()
{
while(scanf("%d%d",&n,&m)!=EOF)
{
sum[]=;
for(int i=,tmp;i<=n;i++)
{
scanf("%d",&tmp);
sum[i]=sum[i-]+tmp;
} dp[]=;
head=tail=;
q[tail++]=;
for(int i=;i<=n;i++)
{
//计算dp[i]
while(tail-head>= && gUp(q[head],q[head+])<=sum[i]*gDown(q[head],q[head+]))
head++;
dp[i]=getDP(i,q[head]); //入队点i
while(tail-head>= && gUp(q[tail-],q[tail-])*gDown(q[tail-],i)>=gUp(q[tail-],i)*gDown(q[tail-],q[tail-]))
tail--;
q[tail++]=i;
} printf("%d\n",dp[n]);
}
}

有一点需要注意的是,本题中c[i]其实是有可能等于0的,所以不能直接gUp/gDown,而是要把分母gDown乘到其另一边再进行比较。

刚开始头铁,强行用list来模拟本题的队列,写*Q.begin()和*++Q.begin()和*Q.end()和*--Q.end()丑的一B,

强行写完之后,发现因为耗时O(size)的size()方法超时,定义一个_size变量代替之后发现确实还是不如直接数组模拟快,

更严重的是,用list写法并没有简单好看多少,最后还是老老实实数组模拟队列吧……

HDU 3507 - Print Article - [斜率DP]的更多相关文章

  1. hdu 3507 Print Article(斜率优化DP)

    题目链接:hdu 3507 Print Article 题意: 每个字有一个值,现在让你分成k段打印,每段打印需要消耗的值用那个公式计算,现在让你求最小值 题解: 设dp[i]表示前i个字符需要消耗的 ...

  2. HDU 3507 Print Article(DP+斜率优化)

     Print Article Time Limit: 9000/3000 MS (Java/Others)    Memory Limit: 131072/65536 K (Java/Others) ...

  3. HDU 3507 Print Article 斜率优化

    Print Article Time Limit: 9000/3000 MS (Java/Others)    Memory Limit: 131072/65536 K (Java/Others)To ...

  4. hdu 3507 Print Article —— 斜率优化DP

    题目:http://acm.hdu.edu.cn/showproblem.php?pid=3507 设 f[i],则 f[i] = f[j] + (s[i]-s[j])*(s[i]-s[j]) + m ...

  5. DP(斜率优化):HDU 3507 Print Article

    Print Article Time Limit: 9000/3000 MS (Java/Others)    Memory Limit: 131072/65536 K (Java/Others)To ...

  6. HDU 3507 Print Article(斜率优化DP)

    题目链接 题意 : 一篇文章有n个单词,如果每行打印k个单词,那这行的花费是,问你怎么安排能够得到最小花费,输出最小花费. 思路 : 一开始想的简单了以为是背包,后来才知道是斜率优化DP,然后看了网上 ...

  7. ●HDU 3507 Print Article

    题链: http://acm.hdu.edu.cn/showproblem.php?pid=3507 题解: 斜率优化DP 一个入门题,就不给题解了,网上的好讲解很多的.   这里就只提一个小问题吧( ...

  8. HDU 3507 [Print Article]DP斜率优化

    题目大意 给定一个长度为\(n(n \leqslant 500000)\)的数列,将其分割为连续的若干份,使得 $ \sum ((\sum_{i=j}^kC_i) +M) $ 最小.其中\(C_i\) ...

  9. HDU 3507 Print Article(CDQ分治+分治DP)

    [题目链接] http://acm.hdu.edu.cn/showproblem.php?pid=3507 [题目大意] 将长度为n的数列分段,最小化每段和的平方和. [题解] 根据题目很容易得到dp ...

随机推荐

  1. 5 -- Hibernate的基本用法 -- 要点

    Hibernate的基本用法 ⊙ ORM的基本知识 ⊙ ORM和Hibernate的关系 ⊙ Hibernate的基本映射思想 ⊙ Hibernate入门知识 ⊙ 使用Eclipse开发Hiberna ...

  2. 【Android】录音-amr音频录制

    http://www.cnblogs.com/fengzhblog/archive/2013/08/01/3231500.html http://blog.csdn.net/fan7983377/ar ...

  3. [ZZ]c++ cout 格式化输出浮点数、整数及格式化方法

    C语言里可以用printf(),%f来实现浮点数的格式化输出,用cout呢...?下面的方法是在网上找到的,如果各位有别的办法谢谢留下... iomanip.h是I/O流控制头文件,就像C里面的格式化 ...

  4. HTML 样式

    style 属性用于改变 HTML 元素的样式,常见的样式如下: 定义字体颜色:style="color:red"定义字体大小:style="font-size:20px ...

  5. C#中的垃圾回收机制与delegate

    在DeepStream的C#版本调试过程中,发现了一个问题,运行一段时间后,大概每次内存到16M(Debug模式)就会异常 错误“System.NullReferenceException:未将对象引 ...

  6. Ansible的Playbook的编写

    在Ansible中,将各个模块组合起来成为一个YAML格式的配置文件,这个配置文件叫做Playbook, Playbook和模块的关系类似于shell脚本和Linux命令之间的关系. Playbook ...

  7. [SublimeText] Sublime Text 2 在 Ubuntu 上安装指南

    1. 下载Sublime Text 2 在官网下载对应系统位数的版本,从压缩包中提取出源代码,解压后文件夹中的"sublime_text"双击即可直接运行. 2. 建立快捷链接 将 ...

  8. React Native(五)——获取设备信息react-native-device-info

    心酸史: 自从接触rn开始后,越来越多的引入第三方组件而开始的配置文件,让自己一再头疼: 明明是按照官方文档一步一步的配置,为什么别人可以做到的自己却屡屡出错,真是哭笑不得--从微信分享react-n ...

  9. STL——配接器(adapters)

    一.配接器 <Design Patterns>一书提到23个最普及的设计模式,其中对adapter样式的定义如下:将一个class的接口转换为另一个class 的接口,使原本因接口不兼容而 ...

  10. 简单的Excel导入(上传、解析、持久化)

    /** * excel导入 * @param req * @param resp * @return */ public void excelImport(){ //先将要上传的Excel文件上传到项 ...