本题解用于本蒟蒻加深算法印象,也欢迎大家阅读


本篇题解将分为四块,一步一步地讲解本题,

Part 1: O(n^3)

\(n^3\) 算法应该非常的显然,我们设 \(f_{i,j}\) 为到 \(i\) 个任务,分为 \(j\) 个批次所使用的最少时间,易得转移方程为:

\[\begin{array}{c}
f_{i,j}=min(f_{k,j-1}+(sumt_i+s*j)*(suma_i-suna_k))\\
(0\le k<i)
\end{array}
\]

但是这个复杂很明显是过不了的( \(O(n^2)\) 都过不了,这还想过???),所以我们考虑优化。

Part 2: O(n^2)

因为每一个前面的 \(s\) 是每次会对后面的动态规划产生影响的,所以我们考虑在设计状态的时候就将 \(s\) 考虑进去,即 \(f_i\) 表示到第 \(i\) 个点,前面 \(i\) 个点的实际用时与 \(s\) 对于后面的影响值的和最小值,这个思想叫做费用提前,我们可以结合状态转移方程来一起食用:

\[\begin{array}{c}
f_i=min(f_j+sumt_i*(suma_i-suma_j)+s*(suma_n-suma_j)\\
(0\le j<i)
\end{array}
\]

看上去好像没什么问题,最优子结构是成立的,这个复杂度是 \(O(n^2)\) 。

Part 3: O(n)

来到今天的重头戏——斜率优化

我们可以先看看这题的弱化版。转移方程和题目大意是一模一样的,唯一的不同点是 \(t_i\) 是非负的。

对于上面的状态的转移方程,我们进行一些移项:

\[\begin{array}{c}
f_i=min(f_j+sumt_i*(suma_i-suma_j)+s*(suma_n-suma_j)\\
\\
f_i=f_j+sumt_i*suma_i-sumt_i*suma_j+s*suma_n-s*suma_j\\
\\
f_j=(sumt_i+s)*suma_j+f_i-sumt_i*suma_i-s*suma_n
\end{array}
\]

此时,如果我们将 \(suma_j\) 看作自变量, \(f_j\) 看作因变量,对于每一个 \(i\) , \(sumt_i\),\(suma_i\),\(suma_n\),\(s\)又都是定值,我们可以将这个式子看作一个待定 \(f_i\) 值的函数。

对于这个函数,我们要使得 \(f_i\) 的数值最小,就是使得函数的截距最小(因为其他的都是定植),而如何找到使得函数截距最小的点呢?我们来画一下图:

可以发现,我们将这个函数的直线向上移动的时候,最先碰到的一个点就是满足条件的点,而我们如何维护这么一个点呢,我们可以发现一些可能的点的性质:

  1. 任意两个可能的点的连线下侧不能有点存在

  2. 所有可能被选中的点构成的集合中,相邻的两个点的斜率是单调递增的,即下凸壳。

大家可以结合图像理解理解:

因为这个函数的斜率 \(sumt_i+s\) 是单调递增的,因为 \(t_i\) 是非负的,所以我们可以发现,对于已经放弃的点,最后也不可能被重新启用。

而每进入一个点时,由于 \(suma_i\) 是单调递增的(因为 \(a_i\) 是非负的),所以我们是向着 \(x\) 轴正方向更新点的,根据性质 \(1\) ,如果这个点向当前最右点连线后,有点出现在直线的下方,则说明最右点不成立,而之后也不可能成立。

结合上面两种操作和性质 \(2\) ,我们可以想到用单调队列维护,维护方式就是上面的操作。

代码如下:

while(head<tail&&cal(q[head],q[head+1])<=sumt[i]+s)
++head;
f[i]=f[q[head]]+sumt[i]*(suma[i]-suma[q[head]])+s*(suma[n]-suma[q[head]]);
while(head<tail&&cal(q[tail-1],q[tail])>=cal(q[tail-1],i))
--tail;
q[++tail]=i;

Part 4: O(nlogn)

而针对这一道题目,我们可以发现 \(t_i\) 是有负的,所以我们第一操作就必须放弃了,因为函数的斜率并非是单调递增的。

但是我们依旧可以维护下凸壳的单调性,而对于当前遍历到的函数斜率,我们可以通过二分的方式找到符合条件的点,即该点与其左侧点的连线斜率小于函数斜率,该点与其右侧殿的连线的斜率大于函数斜率。

代码如下:

int l=head,r=tail-1,mid,res=q[tail];
while(l<=r)
{
mid=(l+r)>>1;
if(cal(q[mid],q[mid+1])>sumt[i]+s)
{
res=q[mid];
r=mid-1;
}
else
l=mid+1;
}
f[i]=f[res]+sumt[i]*(suma[i]-suma[res])+s*(suma[n]-suma[res]);
while(head<tail&&cal(q[tail-1],q[tail])>=cal(q[tail-1],i))
--tail;
q[++tail]=i;

总结

我们可以发现,得出斜率优化函数的方法是提取出关于 \(j\) 项,只要能拆出关于 \(j\) 的一次项,剩下的项塞到等号的另一边,就可以进行斜率优化了。

P5785 [SDOI2012]任务安排的更多相关文章

  1. 洛谷 P5785 [SDOI2012] 任务安排

    链接: P5785 弱化版:P2365 题意: 有 \(n\) 个任务待完成,每个任务有一个完成时间 \(t_i\) 和费用系数 \(f_i\),相邻的任务可以被分成一批.从零时刻开始这些任务会被机器 ...

  2. BZOJ 2726: [SDOI2012]任务安排( dp + cdq分治 )

    考虑每批任务对后面任务都有贡献, dp(i) = min( dp(j) + F(i) * (T(i) - T(j) + S) ) (i < j <= N)  F, T均为后缀和. 与j有关 ...

  3. [bzoj P2726] [SDOI2012]任务安排

    [bzoj P2726] [SDOI2012]任务安排 Time Limit: 10 Sec Memory Limit: 128 MB Submit: 1204 Solved: 349[Submit] ...

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

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

  5. 【BZOJ2726】[SDOI2012]任务安排 斜率优化+cdq分治

    [BZOJ2726][SDOI2012]任务安排 Description 机器上有N个需要处理的任务,它们构成了一个序列.这些任务被标号为1到N,因此序列的排列为1,2,3...N.这N个任务被分成若 ...

  6. [BZOJ2726][SDOI2012]任务安排(DP+凸壳二分)

    2726: [SDOI2012]任务安排 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1580  Solved: 466[Submit][Statu ...

  7. BZOJ_2726_[SDOI2012]任务安排_斜率优化+二分

    BZOJ_2726_[SDOI2012]任务安排_斜率优化+二分 Description 机器上有N个需要处理的任务,它们构成了一个序列.这些任务被标号为1到N,因此序列的排列为1,2,3...N.这 ...

  8. 笔记-[SDOI2012]任务安排

    笔记-[SDOI2012]任务安排 [SDOI2012]任务安排 \(f_i\) 表示分配到第 \(i\) 个任务的最小费用. 令 \(st_i=\sum_{h=1}^iT_h\),\(sc_i=\s ...

  9. bzoj 2726: [SDOI2012]任务安排

    Description 机 器上有N个需要处理的任务,它们构成了一个序列.这些任务被标号为1到N,因此序列的排列为1,2,3...N.这N个任务被分成若干批,每批包含相邻的 若干任务.从时刻0开始,这 ...

随机推荐

  1. JS 实现飞机大战

    这是JS版本的飞机大战,和C#版本的思路相同,就是语言上有差别,用来巩固知识.可以将代码直接引入到HTML中就可以看到效果 //编写背景对象 function Background(width,hei ...

  2. Ceph如何实现文件系统的横向扩展

    前言 在跟一个朋友聊天的时候,聊到一个技术问题,他们的一个环境上面小文件巨多,是我目前知道的集群里面规模算非常大的了,但是目前有个问题,一方面会进行一倍的硬件的扩容,而文件的数量也在剧烈的增长着,所以 ...

  3. flink1.10版本StreamGraph生成过程分析

    1.StreamGraph本质 本质就是按照用程序代码的执行顺序构建出来的用于向执行环境传输的流式图,并且可以支持可视化展示给用户的一种数据结构. 2.StreamGraph.StreamNode和S ...

  4. docker 国内源切换加速

    阿里云比较好: 地址: https://cr.console.aliyun.com/cn-hangzhou/instances/mirrors

  5. 4.1 Spring源码 --- 监听器的原理

    目标: 1. 监听器如何使用 2. 监听器的原理 3. 监听器的类型 4. 多播器的概念和作用 5. 接口类型的监听器是如何注册的? 6. 注解类型的监听器和如何注册的? 7. 如果想在所有的bean ...

  6. 阿里面试官:小伙子,你给我说一下Spring Bean初始化的几种常规方式吧

    前言 通过构造方法实例化通过静态工厂实例化通过实例工厂实例化通过FactoryBean实例化 RumenzA实体类 package com.rumenz; public class RumenzA { ...

  7. 你也想当流量UP主?那就点开看看吧!

    2009年6月份,哔哩哔哩(B站)在一众期待中诞生,它汇聚了天南海北当时小众的二次元同好,它也存在诸多不足,大家亲切地叫它"小破站". 而如今,它成长为一棵枝繁叶茂的参天大树,成为 ...

  8. 思维导图MindManager有新手引导功能吗

    无论是对于初次使用Mindmanager思维导图软件的新手来说,还是对于有一定软件使用基础的进阶者来说,Mindmanager思维导图软件的帮助功能都能给予用户很大的指导作用. Mindmanager ...

  9. Folx的分类标签规则怎么自定义

    一个全新的标签,没有任何对应的标签规则,只是一个空有躯壳没有灵魂的标签,是无法用于文件自动分类的,那么如何根据大家的自身需求,创建一个相对应的标签分类规则呢? 下面小编将使用Folx 5版本为大家讲解 ...

  10. 一篇文章彻底搞懂Java的大Class到底是什么

    作者在之前工作中,面试过很多求职者,发现有很多面试者对Java的 Class 搞不明白,理解的不到位,一知半解,一到用的时候,就不太会用. 因为自己本身以前刚学安卓的时候,甚至做安卓2,3年后,也是对 ...