——以此博客来悼念我在\(QBXT\)懵逼的时光


\(rqy\; tql\) (日常%\(rqy\))


概念及用途

斜率优化是\(DP\)的一种较为常用的优化(据说在高中课本里稍有提及),它可以用于优化这样的一种\(DP\)式子
\[dp[i]=a[i]+\max(y_j-k_ix_j)\;\;\; j\in[1,i-1]\]

原理

以下均以上面的\(DP\)方程为例

如果我们将上式中的\((x_j,y_j)\)画到坐标系里,然后画一条斜率为\(k_i\)的直线,则这条直线为的方程为\(y=k_ix+(y_j-k_ix_j)\),所以\(y_j-k_ix_j\)这一部分就是该直线的截距。所以在\(k_i\)确定的情况下,我们要求一个最大值,即该直线的斜率最大,那就相当于要找直线从上向下平移,所碰到的第一个点,这时,它的截距是最大的。

如果我们把这样的点筛出来,就能大大优化时间复杂度。
gif示意图

最后筛出来的点就是图中最后出现的红色点

当两个点在候选队列里,此时再加进来一个点,如果是这样的一种情况:

那我们可以把中间的那个点删去,因为删去它不会影响我们的最优解,他的截距显然要更大一些

如果是这样的一种情况

则三个点暂时全部保留,因为在某些时刻,它们都可能成为最优解

实现方式

我们可以用双端队列来处理,删掉不优的,加进去暂时较优的。在这只发一下代码中比较重要的部分吧

bool check(int a,int b,int c){
    return (y(a)-y(c))*(x(b)-x(c))-(y(b)-y(c))*(x(a)-x(c))>=0;
}
/*
省略中间部分
*/
for(int i=1;i<=n;i++){
        while(h<t&&y(q[h])-k(i)*x(q[h])<y(q[h+1])-k(i)*x(q[h+1])) h++;
        dp[i]=(sum[i]-l)*(sum[i]-l)+y(q[h])-k(i)*x(q[h]);
        while(h<t&&check(q[t-1],q[t],i)) t--;
        q[++t]=i;
    }

代码中的\(x\)函数、\(y\)函数、\(k\)函数、\(check\)函数请按需要更改,模板差不多就是这样了

例题

HNOI2008 玩具装箱TOY

若果我们在输入时将所有的\(C_i\)和\(L+1\),原公式可以化为\((S-L)^2\)

用\(f_i\)表示前\(i\)个数的最小花费
\(f_i=min(f_j+(S_i-S_j-L)^2)\;\; 1 {<=}j{<=}i-1\)
\(f_i=min(f_j+(S_i-L)^2-2(S_i-L)S_j+S_j^2)\)
\(\;\;\;\;=(S_i-L)^2+min(y_j-k_ix_j)\)
\(y_j=f_j+S_j^2,k_i=2(S_i-L),x_j=S_j\)

所以我们就可用斜率优化来\(A\)掉这道题了

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
#define ll long long
int read(){
    int k=0,f=1; char c=getchar();
    for(;c<'0'||c>'9';c=getchar())
      if(c=='-') f=-1;
    for(;c>='0'&&c<='9';c=getchar())
      k=(k<<3)+(k<<1)+c-48;
    return k*f;
}
int n;
ll dp[50010],sum[50010],l;
inline ll x(int i){ return sum[i]; }
inline ll y(int i){ return dp[i]+sum[i]*sum[i]; }
inline ll k(int i){ return 2LL*(sum[i]-l); }
inline bool check(int a,int b,int c){
    return (y(a)-y(c))*(x(b)-x(c))-(y(b)-y(c))*(x(a)-x(c))>=0;
}
ll q[100010],h=1,t=1;
int main(){
    n=read(),l=read()+1;
    for(int i=1;i<=n;i++) sum[i]=sum[i-1]+read()+1;
    for(int i=1;i<=n;i++){
        while(h<t&&y(q[h])-k(i)*x(q[h])>=y(q[h+1])-k(i)*x(q[h+1])) h++;
        dp[i]=(sum[i]-l)*(sum[i]-l)+y(q[h])-k(i)*x(q[h]);
        while(h<t&&check(q[t-1],q[t],i)) t--;
        q[++t]=i;
    }
    cout<<dp[n];
    return 0;
}

DP玄学优化——斜率优化的更多相关文章

  1. 牛客多校第10场J Wood Processing 分治优化/斜率优化 DP

    题意:你有n块木头,每块木头有一个高h和宽w,你可以把高度相同的木头合并成一块木头.你可以选择一些木头消去它们的一部分,浪费的部分是 消去部分的高度 * 木头的宽度,问把n块木头变成恰好m块木头至少要 ...

  2. 【刷题笔记】DP优化-斜率优化

    斜率优化,是一种利用斜率的优化(废话) 关于数论:咕咕咕 部分内容参考自学长 如果有这样的一个状态转移方程: \[f[i]=\min\limits_{j=L_j}^{R_j}\{f[j]+val(j, ...

  3. 「DP 浅析」斜率优化

    #0.0 屑在前面 将结合经典例题 「HNOI2008」玩具装箱 以及 「NOI2007」货币兑换 进行讲解. #1.0 简述 #1.1 适用情况 斜率优化一般适用于状态转移方程如下的 DP \[f_ ...

  4. DP单调队列--斜率优化P3195

    题意:https://www.luogu.com.cn/problem/P3195 思路:https://www.luogu.com.cn/problemnew/solution/P3195 #def ...

  5. 【学习笔记】动态规划—斜率优化DP(超详细)

    [学习笔记]动态规划-斜率优化DP(超详细) [前言] 第一次写这么长的文章. 写完后感觉对斜优的理解又加深了一些. 斜优通常与决策单调性同时出现.可以说决策单调性是斜率优化的前提. 斜率优化 \(D ...

  6. 【BZOJ-1911】特别行动队 DP + 斜率优化

    1911: [Apio2010]特别行动队 Time Limit: 4 Sec  Memory Limit: 64 MBSubmit: 3478  Solved: 1586[Submit][Statu ...

  7. [luogu3628][bzoj1911][APIO2010]特别行动队【动态规划+斜率优化DP】

    题目描述 给你一个数列,让你将这个数列分成若干段,使其每一段的和的\(a \times sum^2 + b \times sum + c\)的总和最大. 分析 算是一道斜率优化的入门题. 首先肯定是考 ...

  8. BZOJ 3437 小P的牧场(斜率优化DP)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=3437 [题目大意] n个牧场排成一行,需要在某些牧场上面建立控制站, 每个牧场上只能建 ...

  9. HDOJ 1300 Pearls 斜率优化dp

    原题连接:http://acm.hdu.edu.cn/showproblem.php?pid=1300 题意: 题目太长了..自己看吧 题解: 看懂题目,就会发现这是个傻逼dp题,斜率优化一下就好 代 ...

随机推荐

  1. Inside Geometry Instancing(上)

    Inside Geometry Instancing(上) http://blog.csdn.net/soilwork/article/details/598335 翻译:claymanclayman ...

  2. WPF 设置 Label.MouseDoubleClickEvent

    facLabel.AddHandler(Label.MouseDoubleClickEvent, new MouseButtonEventHandler((o, a) => { facLabel ...

  3. css 的继承性

    目录 css 的继承性是什么? 父元素的属性那些可以被子元素继承,哪些不能呢? css 的继承性是什么? 在面向对象语言都会存在继承的概念,在面向对象语言中,继承的特点:继承了父类的属性和方法. 那么 ...

  4. 笔记-JavaWeb学习之旅11

    请求转发:一种在服务器内部的资源跳转方式 使用步骤 1.通过request对象获取请求转发器对象:RequestDispatcher getRequestDispatcher(String path) ...

  5. 01 | VIM基础攻略

    启动 vim 后,vim 处于 normal 模式. Step One: "i" -> insert 模式, ESC -> normal 模式: "x&quo ...

  6. VC 中TEXT、_T、L的区别

    http://i.cnblogs.com/EditPosts.aspx?opt=1 对于从VC++6.0转到VS2005编译环境中的程序员.往往会碰到字符集之间的转换. VC6.0采用的是ANSI字符 ...

  7. python 基础(十三) time模块

    日期和时间 一.time模块 import time 时间戳: 时间戳是指格林威治时间1970年1月1日0时0分0秒至现在的秒数   s(秒).ms(毫秒).μs(微秒).ns(纳秒), ​ 其中:1 ...

  8. mysql(MySQL客户端连接工具)

    在MySQL提供的工具中,DBA使用最频繁的莫过于mysql.这里的mysql不是指MySQL服务,也不是mysql数据库,而是连接数据库的客户端工具.类似于Oracle的sqlplus. 语法: m ...

  9. MySQL简单的确定瓶颈

    如果接到报警可能需要ssh看看瓶颈是什么,怎么下手 确定os层 确定磁盘是否够用的:df –h 再看看系统整体状态: top 哪些进程占用资源比较多,能杀就杀 系统的负载 vmstat看看wa值,r列 ...

  10. JavaScript之执行环境及作用域

        执行环境定义了变量或函数有权访问的其他数据,决定了它们各自的行为.每个执行环境都有一个与之关联的变量对象,环境中定义的所有变量和函数都保存在这个对象中.我们编写的代码是无法访问这个对象的,但解 ...