BZOJ 1010 [HNOI2008]玩具装箱 (斜率优化DP)
题目链接
http://www.lydsy.com/JudgeOnline/problem.php?id=1010
思路
【斜率优化DP】
我们知道,有些DP方程可以转化成DP[i]=f[j]+x[i]的形式,其中f[j]中保存了只与j相关的量。这样的DP方程我们可以用单调队列进行优化,从而使得O(n^2)的复杂度降到O(n)。
可是并不是所有的方程都可以转化成上面的形式,比如dp[i]=dp[j]+(x[i]-x[j])*(x[i]-x[j])。如果把右边的乘法化开的话,会得到x[i]*x[j]的项。这就没办法使得f[j]里只存在于j相关的量了。于是上面的单调队列优化方法就不好使了,所以我们需要一种新的优化方法,叫做斜率优化。
拿本题来说,我们设dp[i]表示装到第i个玩具的时候最少的花费,sum[i]表示前i个玩具的长度和。于是方程就是:
dp[i] = min{ dp[j] + [i+sum[i]-(j+1+sum[j])-L]2 };
题目的N<=50000,二维铁定超时了。我们就来看看斜率优化如何做到从O(n^2)复杂度降到O(n)。
分析:
我们假设k<j<i。如果在j的时候决策要比在k的时候决策好,那么也是就是dp[j] + [i+sum[i]-(j+1+sum[j])-L]2 < dp[k] + [i+sum[i]-(j+1+sum[k])-L]2。(因为是最小花费,所以优就是小于)
两边移项一下得到:[dp[j]+(j+1+sum[j])^2-(dp[k]+(k+1+sum[k])^2)]/(2*( (j+1+sum[j])-(k+1+sum[k]) )) < (i + sum[i]-L)。我们把dp[j]-num[j]^2看做是yj,把2*num[j]看成是xj。
那么不就是(yj-yk)/(xj-xk) < A[i]么? 左边是不是斜率的表示?
那么(yj-yk)/(xj-xk) < A[i]说明了什么呢? 我们前面是假设j的决策比k的决策要好才得到这个表示的,那么g[j,k]=(yj-yk)/(xj-xk) < A[i]代表这j的决策比k的决策要更优。
斜率优化在于:①设k<j<i,如果g[i,j] < g[j,k],那么j点便永远不可能成为最优解,可以直接将它踢出我们的最优解集。‘
为什么呢?我们假设g[i,j] < A[i],那么就是说i点要比j点优,排除j点。如果g[i,j] >= A[i],那么j点此时是比i点要更优,但是同时g[j,k]>g[i,j]>A[i]。这说明还有k点会比j点更优,同样排除j点。排除多余的点,这便是一种优化!
由于我们排除了g[i,j] < g[j,k]的情况,所以整个有效点集呈现一种下凸性质,即k j的斜率要小于j i的斜率。这样,从左到右的斜率之间就是单调递增的了,所以我们就可以对g维护一个单调队列。
同时,函数A[i]也要具有单调递增的特性,所以第二个优化就在于:②如果单调队列的头两个点为i, j,斜率g[j, i] < A[i],则说明j优于i,并且由于A单调递增,所以g[j,i]恒小于A[],所以可以直接把i排除掉。
于是对于这类题斜率优化做法可以总结如下:
1,用一个单调队列来维护解集。
2,假设队列中从头到尾已经有元素a b c。那么当d要入队的时候,我们维护队列的下凸性质,即如果g[d,c]<g[c language=",b"][/c],那么就将c点删除。直到找到g[d,x]>=g[x,y]为止,并将d点加入在该位置中。
3,求解时候,从队头开始,如果已有元素a b c,当i点要求解时,如果g[b,a]<A[i],那么说明b点比a点更优,a点可以排除,于是a出队。最后dp[i]=getDp(q[head])。
代码
[cpp]
#include <iostream>
#include <cstdio>
#include <cmath>
#include <algorithm>
#include <string>
#include <cstring>
#include <vector>
#include <queue>
#define MID(x,y) ((x+y)/2)
#define MEM(a,b) memset(a,b,sizeof(a))
#define REP(i, begin, end) for (int i = begin; i <= end; i ++)
using namespace std;
long long dp[50005], sum[50005];
vector <int> Q;
int n, L;
inline long long getDP(int i, int j){
return dp[j] + (i+sum[i]-(j+1+sum[j])-L) * (i+sum[i]-(j+1+sum[j])-L);
}
inline double getUP(int j, int k){
return (double)dp[j] + (double)(j+1+sum[j]) * (double)(j+1+sum[j]) - ((double)dp[k] + (double)(k+1+sum[k]) * (double)(k+1+sum[k]));
}
inline double getDOWN(int j, int k){
return 2.0 * ((j+1+sum[j]) - (k+1+sum[k]));
}
inline double getRIGHT(int i){
return (double)(sum[i] + i - L);
}
inline double slope(int j, int k){
return getUP(j, k) / getDOWN(j, k);
}
int main(){
scanf("%d %d", &n, &L);
sum[0] = 0, dp[0] = 0;
for (int i = 1; i <= n; i ++){
scanf("%lld", &sum[i]);
sum[i] += sum[i-1];
}
Q.clear();
Q.push_back(0);
for (int i = 1; i <= n; i ++){
while((int)Q.size() > 1 && slope(Q[1], Q[0]) <= getRIGHT(i))
Q.erase(Q.begin());
int tmp = Q[0];
dp[i] = getDP(i, Q[0]);
while( (int)Q.size() > 1 && slope(Q[(int)Q.size()-1], Q[(int)Q.size()-2]) >= slope(i, Q[(int)Q.size()-1]) )
Q.pop_back();
Q.push_back(i);
}
printf("%lld\n", dp[n]);
return 0;
}
[/cpp]
BZOJ 1010 [HNOI2008]玩具装箱 (斜率优化DP)的更多相关文章
- BZOJ 1010 HNOI2008 玩具装箱 斜率优化
题目链接: http://www.lydsy.com/JudgeOnline/problem.php?id=1010 Description P教授要去看奥运,但是他舍不下他的玩具,于是他决定把所有的 ...
- bzoj 1010 [HNOI2008]玩具装箱toy(DP的斜率优化)
1010: [HNOI2008]玩具装箱toy Time Limit: 1 Sec Memory Limit: 162 MBSubmit: 7874 Solved: 3047[Submit][St ...
- BZOJ 1010: [HNOI2008]玩具装箱toy [DP 斜率优化]
1010: [HNOI2008]玩具装箱toy Time Limit: 1 Sec Memory Limit: 162 MBSubmit: 9812 Solved: 3978[Submit][St ...
- BZOJ 1010: [HNOI2008]玩具装箱toy 斜率优化DP
1010: [HNOI2008]玩具装箱toy Description P教授要去看奥运,但是他舍不下他的玩具,于是他决定把所有的玩具运到北京.他使用自己的压缩器进行压缩,其可以将任意物品变成一堆,再 ...
- Bzoj 1010: [HNOI2008]玩具装箱toy(斜率优化)
1010: [HNOI2008]玩具装箱toy Time Limit: 1 Sec Memory Limit: 162 MB Description P教授要去看奥运,但是他舍不下他的玩具,于是他决定 ...
- BZOJ 1010 [HNOI2008]玩具装箱toy
1010: [HNOI2008]玩具装箱toy Time Limit: 1 Sec Memory Limit: 162 MBSubmit: 7184 Solved: 2724[Submit][St ...
- BZOJ 1010: [HNOI2008]玩具装箱toy(斜率优化dp)
http://www.lydsy.com/JudgeOnline/problem.php?id=1010 题意: 思路: 容易得到朴素的递归方程:$dp(i)=min(dp(i),dp(k)+(i-k ...
- BZOJ 1010 [HNOI2008]玩具装箱toy:斜率优化dp
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1010 题意: 有n条线段,长度分别为C[i]. 你需要将所有的线段分成若干组,每组中线段的 ...
- BZOJ.1010.[HNOI2008]玩具装箱toy(DP 斜率优化/单调队列 决策单调性)
题目链接 斜率优化 不说了 网上很多 这的比较详细->Click Here or Here //1700kb 60ms #include<cstdio> #include<cc ...
随机推荐
- Flask form(登录,注册)
用户登录 from flask import Flask, render_template, request, redirect from wtforms import Form from wtfor ...
- js null, undefined, NaN, ‘’, false, 0, ==, === 全验证
<html> <head> <meta charset="utf-8" /> </head> <body> <in ...
- Log4net 自定义字段 写入Oracle 使用ODP.NET Managed驱动
一.环境说明: 开发工具:vs2010 ,数据库:oracle 11g ,版本:log4net的目前最新版本1.2.13.0 : Oracle.ManagedDataAccess.dll ...
- Straight Master Gym-101775J (思维+差分)
题意:给出N种类的数量,求是否可以把N种牌按3-5张连续的顺子打出,顺子必须连续. 分析:相当于把这个序列分成若干长度为[3,5]的区间,当然其实分成若干段大于3的区间即可.因为大于5的区间又可以分拆 ...
- Winter-2-STL-B Brackets 解题报告及测试数据
Time Limit:2000MS Memory Limit:65536KB Description Given a string consisting of brackets of two ...
- ABP官方文档翻译 1.3 模块系统
ABP模块系统 介绍 模块定义 生命周期方法 PreInitialize Initialize PostInitialize Shutdown 模块依赖 插件系统 ASP.NET Core ASP.N ...
- path.resolve()和path.join()
resolve 作用:path.resolve() 该方法将一些的 路径/路径段 解析为绝对路径. 语法:path.resolve([...paths]) 说明: ...paths <strin ...
- JavaScript&jQuery获取url参数方法
JavaScript&jQuery获取url参数方法 function getUrlParam(name){ var reg = new RegExp("(^|&)" ...
- markdown工作随笔总结
1. 锚点 (使用方法和链接很像) ## 目录 1. [命名](#命名) ....... **[返回顶部](#目录)** ## 命名 ###命名原则 可以从返回顶部回到目录,也可以点击目录的命名跳到命 ...
- linux消息队列应用编程
消息队列: 消息队列提供了一个从一个进程向另外一个进程发送一块数据的方法 每个数据块都被认为是有一个类型,接收者进程接收的数据块可以有不同的类型值 消息队列也有管道一样的不足,就是每个消息的 ...