洛谷题目传送门

用两种不一样的思路立体地理解斜率优化,你值得拥有。

题意分析

既然所有的土地都要买,那么我们可以考虑到,如果一块土地的宽和高(其实是蒟蒻把长方形立在了平面上)都比另一块要小,那么肯定是直接并购,这一块对答案没有任何贡献。

我们先把这些给去掉,具体做法可以是,按高为第一关键字,宽为第二关键字从大到小排序,然后上双指针扫一遍。

于是,剩下的就是一个高度递减、宽度递增的矩形序列。考虑怎样制定它们的并购方案会最优。显然如果要并购,一定要挑序列中的一段区间,这样贡献答案的就只有最左边矩形的高乘上最右边矩形的宽,中间的又是没有贡献了。

设\(f_i\)为前\(i\)个矩形的最小花费,\(w\)为宽,\(h\)为高,直接写出一个\(O(n^2)\)的方程

\[f_i=\min\limits_{j=1}^i\{f_{j-1}+w_ih_j\}
\]

一看貌似是一个决策单调性优化的式子。然而。。。。。。

初中生都会的函数图像法

这种理解方法是在决策单调性优化DP的基础上应运而生的。

或者说,(在大多数情况下)斜率优化可以看作决策单调性优化的一种特殊情形。蒟蒻建议还是先入手决策单调性再来斜率优化吧。

蒟蒻的DP各种优化总结

蒟蒻之前写的一道经典(裸)决策单调性题的题解戳这里(Lightning Conductor)

对于每一个\(f_{j-1}+w_ih_j\),我们都可以把它视为一个直线\(l_j:y=ax+b\),其中\(a=h_j,b=f_{j-1}\)。对于每一个\(i\),我们就是需要求出所有\(j\le i\)的直线的\(x\)取\(w_i\)时最小的一个\(y\)值。仍然用KmPlot画一个我们需要维护的所有直线的样子,它们应该满足斜率依次递减。

\(l_1:y=2x;\)

\(l_2:y=x+1;\)

\(l_3:y=\frac x 2+3;\)

\(l_4:y=\frac x 6+5.\)

真正有用的部分

这样的话,我们就用单调队列维护若干个斜率递减的函数。我们仍然需要按照决策单调性的做法,维护相邻两个决策直线间的临界值(交点)\(k\)。难道还要维护决策二分栈,对每个临界值都二分么?

这些决策不是直线吗?求两个直线的交点。。。。。。初中数学就教了,是\(O(1)\)的。也就是对两个相邻决策直线\(l_1,l_2\),我们求\(\frac{b_2-b_1}{a_1-a_2}\)。其它过程跟决策单调性是一模一样的。直线入队前,如果队尾不满足斜率递增则出队。求\(f_i\)之前,先把队首临界值\(\le w_i\)的决策出队,那么现在队首就是最优决策了。

这样求出\(f_n\)只需要\(O(n)\)的时间。

高中生都会的线性规划法

这才是理解斜率优化的正宗方法,因为上面并没有充分体现对斜率的处理过程。

上面对两个相邻直线求\(\frac{b_2-b_1}{a_1-a_2}\),看起来有点像求什么东西。

我们原来把决策当成直线,斜率为\(a\),截距为\(b\)。现在我们换一下。把决策\(f_{j-1}+w_ih_j\)看作一个点\(p_j(x,y)\),其中\(x=-h_j,y=f_{j-1}\)。

现在要求解的问题又变成了什么样子呢?在平面上有若干个点,把\(f_i\)看成目标函数\(z\),我们需要找到\(f_i=w_ih_j+f_{j-1}\)即\(z=-w_ix+y\)的最小值。这不是个线性规划么?

把式子变成\(y=w_ix+f_i\),现在就让我们来最小化截距\(f_i\)。手(nao)动(dong)模拟一下,我们现在正在拿着一个斜率为\(w_i\)的直线,从下往上移动,当第一次经过某个决策点的时候,直线的在\(y\)轴上的截距就是我们要求的目标函数\(f_i\)的最小值了。

随便画一堆点就可以发现,无论直线以怎样的斜率向上靠,总有一些点永远都不会第一次与直线相交,也就是说这些决策是没用的。剩下的有用的决策点会构成一个凸包:(因为要画点所以换成了GeoGebra)

凸包的性质就是斜率递增/递减。在此题中,因为\(w\)递增,所以我们的单调队列中存的是若干个点满足\(x\)递增(\(h\)递减),\(y\)递增,而且相邻两个点的斜率也递增。这和原序列的顺序是同向的。假设队尾下标为\(t\),当需要在队尾加入一个新的决策点时,我们可能会遇到这样的情况:

这时候\(p_t\)已经不优了,我们把它出队,如此直到满足斜率递增为止,\(p_i\)就可以入队了。和上面那种理解方法的写法差不多,求相邻两个点形成的直线斜率然后比一下大小。队首的处理跟上面那种理解方法的写法也差不多,如果队首与后一个的斜率小于\(w_i\)就出队。最后的队首依然是最优解。

实现

两种实现的代码长得都差不多,都要搞一个单调队列,都要求临界值/斜率。所以就放一个代码吧。。。

复杂度\(O(n\log n)\),瓶颈竟然在sort上?!蒟蒻可不想来什么wys排序

#include<cstdio>
#include<algorithm>
#define RG register
#define R RG int
#define G c=getchar()
#define Calc(i,j) (f[j-1]-f[i-1])/(a[i].h-a[j].h)
//method1:求出临界值
//method2:求出斜率
using namespace std;
const int N=1e5+9;
int q[N];
double f[N],k[N];
//method1:k_i为决策直线q_i与q_i+1的临界值(交点)
//method2:k_i为决策点q_i与q_i+1所连成直线的斜率
struct Land{
int w,h;//结构体排序
inline bool operator<(RG Land&x)const{
return h>x.h||(h==x.h&&w>x.w);
}
}a[N];
inline int in(){
RG char G;
while(c<'-')G;
R x=c&15;G;
while(c>'-')x=x*10+(c&15),G;
return x;
}
int main(){
R n=in(),i,h,t;
for(i=1;i<=n;++i)
a[i].w=in(),a[i].h=in();
sort(a+1,a+n+1);
for(h=i=1;i<=n;++i)//双指针扫描去除无用矩形
if(a[h].w<a[i].w)a[++h]=a[i];
n=h;
for(h=i=1,t=0;i<=n;++i){
while(h<t&&k[t-1]>=Calc(q[t],i))--t;//维护临界值/斜率单调
k[t]=Calc(q[t],i);q[++t]=i;//加入决策直线/决策点
while(h<t&&k[h]<=a[i].w)++h;//弹出已经不优的决策
f[i]=(double)a[q[h]].h*a[i].w+f[q[h]-1];//求出最优解
}
printf("%.0lf\n",f[n]);
return 0;
}

洛谷P2900 [USACO08MAR]土地征用Land Acquisition(动态规划,斜率优化,决策单调性,线性规划,单调队列)的更多相关文章

  1. 洛谷P2900 [USACO08MAR]土地征用Land Acquisition(斜率优化)

    题意 约翰准备扩大他的农场,眼前他正在考虑购买N块长方形的土地.如果约翰单买一块土 地,价格就是土地的面积.但他可以选择并购一组土地,并购的价格为这些土地中最大的长 乘以最大的宽.比如约翰并购一块3 ...

  2. 洛谷2900 [USACO08MAR]土地征用Land Acquisition (斜率优化+dp)

    自闭的一批....为什么斜率优化能这么自闭. 首先看到这个题的第一想法一定是按照一个维度进行排序. 那我们不妨直接按照\(h_i\)排序. 我们令\(dp[i]\)表示到了第\(i\)个矩形的答案是多 ...

  3. 洛谷 P2900 [USACO08MAR]土地征用Land Acquisition 解题报告

    P2900 [USACO08MAR]土地征用Land Acquisition 题目描述 约翰准备扩大他的农场,眼前他正在考虑购买N块长方形的土地.如果约翰单买一块土 地,价格就是土地的面积.但他可以选 ...

  4. 【洛谷 P2900】 [USACO08MAR]土地征用Land Acquisition(斜率优化,单调栈)

    题目链接 双倍经验 设\(H\)表示长,\(W\)表示宽. 若\(H_i<H_j\)且\(W_i<W_j\),显然\(i\)对答案没有贡献. 于是把所有点按\(H\)排序,然后依次加入一个 ...

  5. luogu P2900 [USACO08MAR]土地征用Land Acquisition

    写这道题时,预处理部分少打了等号,吓得我以为斜率优化错了或者被卡精了 mmp 首先有一个很明显的结论(逃),就是一个土地如果长(\(x\))与宽(\(y\))都比另一个土地小,那么这个土地一定可以跟那 ...

  6. P2900 [USACO08MAR]土地征用Land Acquisition

    \(\color{#0066ff}{ 题目描述 }\) 约翰准备扩大他的农场,眼前他正在考虑购买N块长方形的土地.如果约翰单买一块土 地,价格就是土地的面积.但他可以选择并购一组土地,并购的价格为这些 ...

  7. [LuoguP2900] [USACO08MAR]土地征用(Land Acquisition)

    土地征用 (Link) 约翰准备扩大他的农场,眼前他正在考虑购买N块长方形的土地.如果约翰单买一块土 地,价格就是土地的面积.但他可以选择并购一组土地,并购的价格为这些土地中最大的长 乘以最大的宽.比 ...

  8. 不失一般性和快捷性地判定决策单调(洛谷P1912 [NOI2009]诗人小G)(动态规划,决策单调性,单调队列)

    洛谷题目传送门 闲话 看完洛谷larryzhong巨佬的题解,蒟蒻一脸懵逼 如果哪年NOI(放心我这样的蒟蒻是去不了的)又来个决策单调性优化DP,那蒟蒻是不是会看都看不出来直接爆\(0\)?! 还是要 ...

  9. 洛谷P3515 [POI2011]Lightning Conductor(动态规划,决策单调性,单调队列)

    洛谷题目传送门 疯狂%%%几个月前就秒了此题的Tyher巨佬 借着这题总结一下决策单调性优化DP吧.蒟蒻觉得用数形结合的思想能够轻松地理解它. 首先,题目要我们求所有的\(p_i\),那么把式子变一下 ...

随机推荐

  1. ASP.NET Core StaticFiles中间件修改wwwroot(转载)

    ASP.NET Core 开发,中间件(StaticFiles)的使用,我们开发一款简易的静态文件服务器.告别需要使用文件,又需要安装一个web服务器.现在随时随地打开程序即可使用,跨平台,方便快捷. ...

  2. 【LeetCode106】Construct Binary Tree from Inorder and Postorder Traversal★★

    1.题目 2.思路 思路和LeetCode105类似,见上篇. 3.java代码 //测试 public class BuildTreeUsingInorderAndPostorder { publi ...

  3. Luogu4219 BJOI2014 大融合 LCT

    传送门 题意:写一个数据结构,支持图上连边(保证图是森林)和询问一条边两端的连通块大小的乘积.$\text{点数.询问数} \leq 10^5$ 图上连边,$LCT$跑不掉 支持子树$size$有点麻 ...

  4. 请允许我转载一篇关于套接字的博客:Socket

    这一篇文章,我将图文并茂地介绍Socket编程的基础知识,我相信,如果你按照步骤做完实验,一定可以对Socket编程有更好地理解. 本文源代码,可以通过这里下载 http://files.cnblog ...

  5. Item 9: 比起typedef更偏爱别名声明(alias declaration)

    本文翻译自modern effective C++,由于水平有限,故无法保证翻译完全正确,欢迎指出错误.谢谢! 博客已经迁移到这里啦 我确信我们都同意使用STL容器是一个好主意,并且我希望在Item ...

  6. 【react】使用 create-react-app 构建基于TypeScript的React前端架构----上

    写在前面 一直在探寻,那优雅的美:一直在探寻,那精湛的技巧:一直在探寻,那简单又直白,优雅而美丽的代码. ------ 但是在JavaScript的动态类型.有时尴尬的自动类型转换,以及 “0 == ...

  7. 开启C语言的学习之门

    本人是一枚工业界的码农,为了职业道路越来越宽广决定向上位机方面进军,C语言曾经在大学里面学过点皮毛但是离应用远远不够,尽量每天在工作之余更新自己学习的进度,同时也希望有大神能给予在编程道路上的指导,话 ...

  8. git bash返回上一级目录

    YITU-LIUMZ+Administrator@yitu-liumz MINGW64 ~/learngit/gitskills (dev)$ cd ..\ 注意 cd 后面有空格 然后就会弹出一个 ...

  9. ul ol li的序号编号样式

    序号样式例子,下面是html代码(做参考) <ol> <li>列表内容列表内容列表内容列表</li> <li>列表内容列表内容列表内容列表</li ...

  10. 置换群 Burnside引理 Pólya定理(Polya)

    置换群 设\(N\)表示组合方案集合.如用两种颜色染四个格子,则\(N=\{\{0,0,0,0\},\{0,0,0,1\},\{0,0,1,0\},...,\{1,1,1,1\}\}\),\(|N|= ...