描述

这道题目说的是,给出了n项必须按照顺序完成的任务,每项任务有它需要占用机器的时间和价值。现在我们有一台机器可以使用,它每次可以完成一批任务,完成这批任务所需的时间为一个启动机器的时间S加上所有任务需要的时间。并且它是在完成所有任务后才会把任务的成果输出,这样我们就在那一时间时得到所有这些任务的一个完成时间。我们现在要求一种完成任务的方式使得所有任务的完成时间乘上该任务的价值之和最小。

输入

第一行,一个数n表示任务的总数

第二行,一个数s表示开机的时间

接下来n行,每行两个数a,b,a表示该任务完成所要的时间,b表示该任务的价值。

输出

输出文件包括一行,这一行只包含一个整数,就是最少价值和。

样例输入

5

1

1 3

3 2

4 3

2 3

1 4

样例输出

153

提示

【样例说明】

分组情况:

{1,2},{3},{4,5}

T=(s+1+3)(3+2)+(s+s+1+3+4)3+(s+s+s+1+3+4+2+1)*(3+4)=153

【数据规模】

对于全部的数据,保证有n<=10000。

标签

算法竞赛进阶指南


斜率优化dp入门题。

这道题需要简单的推一推式子。

我们用两个前缀和w[i],time[i]" role="presentation" style="position: relative;">w[i],time[i]w[i],time[i]分别表示价值与时间的前缀和。

用f[i]" role="presentation" style="position: relative;">f[i]f[i]表示前i个物品最优的安排需要的花费。

则有:

f[i]=min(f[j]+S∗(w[n]−w[j])+time[i]∗(w[i]−w[j]))" role="presentation" style="position: relative;">f[i]=min(f[j]+S∗(w[n]−w[j])+time[i]∗(w[i]−w[j]))f[i]=min(f[j]+S∗(w[n]−w[j])+time[i]∗(w[i]−w[j]))

其中我们使用了一种叫做费用提前计算的思想,把当前的启动时间对之后所有任务的影响都计算了出来。

但只推出这个式子每次转移是O(n)" role="presentation" style="position: relative;">O(n)O(n)的,无法匹配这道题的数据范围。

于是我们将min去掉,变成:

=>f[i]=f[j]+S∗(w[n]−w[j])+time[i]∗(w[i]−w[j])" role="presentation" style="position: relative;">f[i]=f[j]+S∗(w[n]−w[j])+time[i]∗(w[i]−w[j])f[i]=f[j]+S∗(w[n]−w[j])+time[i]∗(w[i]−w[j])

然后移个项:

=>f[j]=f[i]−S∗(w[n]−w[j])−time[i]∗(w[i]−w[j])" role="presentation" style="position: relative;">f[j]=f[i]−S∗(w[n]−w[j])−time[i]∗(w[i]−w[j])f[j]=f[i]−S∗(w[n]−w[j])−time[i]∗(w[i]−w[j])

然后展开括号并整理式子:

=>f[j]=(S+time[i])∗w[j]+f[i]−S∗w[n]−time[i]∗w[i]" role="presentation" style="position: relative;">f[j]=(S+time[i])∗w[j]+f[i]−S∗w[n]−time[i]∗w[i]f[j]=(S+time[i])∗w[j]+f[i]−S∗w[n]−time[i]∗w[i]

发现这是一个关于w[j]的一次函数,于是对于每一个j" role="presentation" style="position: relative;">jj我们对应成平面上的点(w[j],f[j])" role="presentation" style="position: relative;">(w[j],f[j])(w[j],f[j]),可以看到w[j]是单调递增的。

对于上面的式子,我们想要f[i]最小,就需要让这个一次函数的截距最小。

显然需要维护一个下凸壳,并且将过于平缓的线段(斜率小于S+time[j]" role="presentation" style="position: relative;">S+time[j]S+time[j]的一定不如后面的线段优)删去就行了。


代码:

#include<bits/stdc++.h>
#define ll long long
#define N 10005
using namespace std;
inline ll read(){
    ll ans=0;
    char ch=getchar();
    while(!isdigit(ch))ch=getchar();
    while(isdigit(ch))ans=(ans<<3)+(ans<<1)+(ch^48),ch=getchar();
    return ans;
}
int n,S,q[N],hd,tl;
ll w[N],tim[N],f[N];
inline double slope(int x,int y){return (f[x]-f[y])/(w[x]-w[y]);}
int main(){
    n=read(),S=read(),hd=1,tl=1;
    for(int i=1;i<=n;++i)tim[i]=read()+tim[i-1],w[i]=read()+w[i-1];
    for(int i=1;i<=n;++i){
        while(hd<tl&&1.0*(S+tim[i])>slope(q[hd+1],q[hd]))++hd;
        f[i]=f[q[hd]]+S*(w[n]-w[q[hd]])+tim[i]*(w[i]-w[q[hd]]);
        while(hd<tl&&slope(q[tl],q[tl-1])>slope(i,q[tl]))--tl;
        q[++tl]=i;
    }
    cout<<f[n];
    return 0;
}

2018.09.05 任务安排(斜率优化dp)的更多相关文章

  1. bzoj 2726 任务安排 斜率优化DP

    这个题目中 斜率优化DP相当于存在一个 y = kx + z 然后给定 n 个对点 (x,y)  然后给你一个k, 要求你维护出这个z最小是多少. 那么对于给定的点来说 我们可以维护出一个下凸壳,因为 ...

  2. [SDOI2012]任务安排 - 斜率优化dp

    虽然以前学过斜率优化dp但是忘得和没学过一样了.就当是重新学了. 题意很简单(反人类),利用费用提前的思想,考虑这一次决策对当前以及对未来的贡献,设 \(f_i\) 为做完前 \(i\) 个任务的贡献 ...

  3. BZOJ 2726: [SDOI2012]任务安排 [斜率优化DP 二分 提前计算代价]

    2726: [SDOI2012]任务安排 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 868  Solved: 236[Submit][Status ...

  4. 2018.09.06 警卫安排(树形dp)

    描述 太平王世子事件后,陆小凤成了皇上特聘的御前一品侍卫. 皇宫以午门为起点,直到后宫嫔妃们的寝宫,呈一棵树的形状:有边直接相连的宫殿可以互相望见.大内保卫森严,三步一岗,五步一哨,每个宫殿都要有人全 ...

  5. BZOJ 2726 [SDOI2012] 任务安排 - 斜率优化dp

    题解 转移方程与我的上一篇题解一样 : $S\times sumC_j  + F_j = sumT_i \times sumC_j + F_i - S \times sumC_N$. 分离成:$S\t ...

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

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

  7. 2018.09.29 bzoj3675: [Apio2014]序列分割(斜率优化dp)

    传送门 斜率优化dp经典题目. 首先需要证明只要选择的K个断点是相同的,那么得到的答案也是相同的. 根据分治的思想,我们只需要证明有两个断点时成立,就能推出K个断点时成立. 我们设两个断点分成的三段连 ...

  8. 2018.09.10 bzoj1597: [Usaco2008 Mar]土地购买(斜率优化dp)

    传送门 终究还是通宵了啊... 这是一道简单的斜率优化dp. 先对所有土地排序,显然如果有严格小于的两块土地不用考虑小的一块. 于是剩下的土地有一条边单增,另外一条单减. 我们假设a[i]是单减的,b ...

  9. 2018.09.07 bzoj1096: [ZJOI2007]仓库建设(斜率优化dp)

    传送门 斜率优化dp经典题. 令f[i]表示i这个地方修建仓库的最优值,那么答案就是f[n]. 用dis[i]表示i到1的距离,sump[i]表示1~i所有工厂的p之和,sum[i]表示1~i所有工厂 ...

随机推荐

  1. java 调用apache.commons.codec的包简单实现MD5加密

    转自:https://blog.csdn.net/mmd1234520/article/details/70210002/ import java.security.MessageDigest; im ...

  2. java.lang.ClassCastException: java.util.Arrays$ArrayList cannot be cast to java.util.ArrayList

    String[] 转换成 ArrayList 报的错. String[] str = {"A","B"}; ArrayList<String> li ...

  3. MySQL命令行学习

    1.登录mysql 本地:mysql -u root -p, 回车后输入密码; 也可以p后不加空格,直接加密码.回车就登录了 远程:mysql -hxx.xx.xx.xx -u -pxxx 2.查看数 ...

  4. 数学公式 AS3应用

    普通做法: var pA:Point=new Point(100,100); var pB:Point=new Point(300,200); var dx:Number=pA.x-pB.x; var ...

  5. PuTTY免输密码自动登录Linux

    1.使用PuTTY安装目录里的puttygen.exe工具.先点“生成(Generate)”,然后随意移动鼠标直到进度条填满,即可生成密钥 公钥部分:把上边那一段文字全选->复制备用.(不要点击 ...

  6. DOM0和D0M2级事件

    1.DOM0级事件:on+事件类型 1.1.在html行内直接绑定, 1.2.在js中绑定 A.DOM0级事件和DOM0级事件相互之间会覆盖,比如以下代码执行后弹出jsDOM0级,js中绑定的事件 覆 ...

  7. ubuntu 安装qq方案

    from: http://blog.csdn.net/fuchaosz/article/details/51919607 安装wineQQ国际版 VirtualBox安装windows,再安装QQ,然 ...

  8. COM组件 IDispatch 及双接口的调用

    转自:http://blog.csdn.net/cnhk1225/article/details/50555647 一.前言 前段时间,由于工作比较忙,没有能及时地写作.其间收到了很多网友的来信询问和 ...

  9. Ubuntu 安装 Zabbix 3.2详细步骤

    创建 zabbix 用户 因为zabbix 程序的守护进程需要非特权用户,所以需要创建一个 zabbix 用户,即使从 root 用户启动 zabbix 程序,也会自动切换到 zabbix 用户,所以 ...

  10. lrzsz的安装与配置

    1)下载http://freshmeat.sourceforge.net/projects/lrzsz/ 2)tar zxvf lrzsz-0.12.20.tar.gz 3)mv lrzsz-0.12 ...