动态规划DP的斜率优化 个人浅解 附HDU 3669 Cross the Wall
首先要感谢叉姐的指导Orz
这一类问题的DP方程都有如下形式
dp[i] = w(i) + max/min(a(i)*b(j) + c(j)) ( 0 <= j < i )
其中,b, c均为j的单调函数。通常情况下a(i)也是单调的,a(i)不单调时就只能二分查找了。
这里讲一下当c(j)单调递增,b(j)单调递减,原方程求min的情况
对于同一j,b(j),c(j)为常数,而a(i)为变量,令y = b(j) * x + c(j),则y为线性函数
我们把( 0 <= j < i )的直线均表示在平面上,则如下图所示。
显然,对于不同的dp[i],我们要求的是在对应x = a(i)下所有直线的最小值
看图我们可以发现,就是这些直线的下半平面交的边缘组成的分段函数,就是我们要求的函数
那么怎么维护这个半平面呢
首先b(j)最小的直线,必然会在足够大处,取得所有直线的最小值
我们在维护的过程中,总是在原半平面交的基础上,加上一个c(j)最大,b(j)最小的直线
这个时候,这条直线必然会加入半平面交当中。
同时,有可能会导致有些直线被遮蔽而退出集合。
我们可以发现,如果j=k的直线没有被遮蔽,那么j=[0,k)的直线也显然不会被遮蔽。
所以我们只需要从原来的半平面交最大的j开始不停的向内找,直到找到第一个没被遮蔽的直线就可以了。
那么直线什么时候会被遮蔽呢?
直线只剩两条的时候,显然不会有直线被遮蔽→ →
观察图中的直线1,2,3以及直线2,3,4。
2没有被遮蔽,是因为2和1的交点在2和3的交点左边。
而3被遮蔽,是因为3和2的交点在3和4的交点的右边。
所以只需要求交点然后比较大小就可以了。(两直线求交,初中知识,相信大家都会,请自行计算)
这里需要注意的是,求交点需要用到除法。但是除法容易出现精度问题,我们要把除法在比较的时候移到另一侧变成乘法,这个时候记得注意正负变号。
这个维护过程,考虑每个直线最多进出队列一次,复杂度总计O(n)
维护好半平面集合,后,求x = a(i)时的值,只需要二分判断落在哪个直线的交点区域内就好了。
特殊的,如果a(i)也单调,则可以像单调队里那样从头部一一剔除的方法,这里不多做介绍。
理解上述部分后,假如b(j),c(j)都不单调,问题则转变为动态半平面交。
另外,其实有没有发现写起来和凸包的理解非常像?叉姐告诉我们,其实根据平面的点与直线的对偶,凸包和半平面交问题是完全等价的。
附HDU 3669代码
#include<cstdio>
#include<cstring>
#include<cmath>
#include<cstdlib>
#include<algorithm>
using namespace std;
typedef long long ll;
const int N=; pair<int,int> a[N],s[N];
int n,m; int l,r;
pair<int,ll> q[N]; int x,y;
ll dp[][N]; inline void push(int k,ll b){
while(l<r && (q[r].second-q[r-].second)*(q[r].first-k)>=(b-q[r].second)*(q[r-].first-q[r].first)) r--;
q[++r]=make_pair(k,b);
} inline void pop(ll x){
while(l<r && (q[l].first - q[l+].first)*x >= q[l+].second - q[l].second) l++;
} int main(){
while(~scanf("%d%d",&n,&m)){
for(int i=;i<n;i++)scanf("%d%d",&a[i].first,&a[i].second);
sort(a,a+n);
int cnt=,mxb=a[n-].second;
s[]=a[n-];
for(int i=n-;i>=;i--)if(a[i].second>mxb){
mxb = a[i].second;
s[cnt++]=a[i];
}
n=cnt;
ll mxa=s[].first;
for(int i=;i<n;i++) dp[][i]=mxa*s[i].second;
ll ans=dp[][n-];
x=;
y=;
if(m>n) m=n;
for(int k=;k<m;k++){
l=;r=-;
push(s[k].first,dp[x][k-]);
for(int i=k;i<n;i++){
pop(s[i].second);
dp[y][i]=q[l].second+(ll)q[l].first*s[i].second;
push(s[i+].first,dp[x][i]);
}
x^=;
y^=;
if(dp[x][n-]<ans) ans=dp[x][n-];
else break;
}
printf("%I64d\n",ans);
}
return ;
}
动态规划DP的斜率优化 个人浅解 附HDU 3669 Cross the Wall的更多相关文章
- hdu 3669 Cross the Wall(斜率优化DP)
题目连接:hdu 3669 Cross the Wall 题意: 现在有一面无限大的墙,现在有n个人,每个人都能看成一个矩形,宽是w,高是h,现在这n个人要通过这面墙,现在只能让你挖k个洞,每个洞不能 ...
- HDU 3669 [Cross the Wall] DP斜率优化
问题分析 首先,如果一个人的\(w\)和\(h\)均小于另一个人,那么这个人显然可以被省略.如果我们将剩下的人按\(w[i]\)递增排序,那么\(h[i]\)就是递减. 之后我们考虑DP. 我们设\( ...
- HDU 3669 Cross the Wall(斜率DP+预处理)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3669 题目大意:有n(n<=50000)个矩形,每个矩形都有高和宽,你可以在墙上最多挖k个洞使得 ...
- dp的斜率优化
对于刷题量我觉得肯定是刷的越多越好(当然这是对时间有很多的人来说. 但是在我看来我的确适合刷题较多的那一类人,应为我对知识的应用能力并不强.这两天学习的内容是dp的斜率优化.当然我是不太会的. 这个博 ...
- 【DP】斜率优化
斜率优化 入门题:PKU3709 很多人貌似都是做这道题来K斜率优化的,所以看了资料以后还是开始入手吧. 然而还是得跪求大神的程序啊 ORZ ORZ…… 其实理解斜率优化就是会列斜率不等式,还要理解剔 ...
- bzoj 1010 [HNOI2008]玩具装箱toy(DP的斜率优化)
1010: [HNOI2008]玩具装箱toy Time Limit: 1 Sec Memory Limit: 162 MBSubmit: 7874 Solved: 3047[Submit][St ...
- 洛谷P3195 [HNOI2008] 玩具装箱 [DP,斜率优化,单调队列优化]
题目传送门 题目描述 P教授要去看奥运,但是他舍不下他的玩具,于是他决定把所有的玩具运到北京.他使用自己的压缩器进行压缩,其可以将任意物品变成一堆,再放到一种特殊的一维容器中.P教授有编号为1...N ...
- 【HDOJ5956】The Elder(树形DP,斜率优化)
题意:有一棵n个点的有根树,每条边上有一个边权.给定P,从i跳到它的祖先j的费用是距离的平方+P,问所有点中到根节点1的总花费最大值 n<=1e5,p<=1e6,w<=1e2 思路: ...
- CF1179D Fedor Runs for President [DP,斜率优化]
Codeforces 思路 考虑把连的那两个点中间的链提出来,那么就会变成一条链,链上的每个点挂着一棵子树的形式. 设那些子树的大小为\(S_1,S2,\cdots\),那么新加的简单路径个数就是 \ ...
随机推荐
- 提高篇(1):RMQ问题与ST表
RMQ是英文Range Minimum/Maximum Query的缩写,是询问某个区间内的最值,这里讲一种解法:ST算法 ST算法通常用在要多次(10^6级别)询问区间最值的问题中,相比于线段树,它 ...
- JavaWeb各大组件生命周期
JavaWeb各大组件生命周期 servlet生命周期 服务器打开:在第一次请求时实例化与初始化:然后进行服务:最后服务器关闭销毁 Cookie生命周期:存储在客户端 如果不设置过期时间,则表示这个c ...
- Centos下使用Docker部署asp.net core项目
本文讲述 CentOS 系统 Docker 中部署 asp.net core开源项目 abp 的过程 步骤 1. 拉取 asp.net core 基础镜像 docker pull microsoft/ ...
- 基于WSAAsyncSelect模型的两台计算机之间的通信
任务目标 编写Win32程序模拟实现基于WSAAsyncSelect模型的两台计算机之间的通信,要求编程实现服务器端与客户端之间双向数据传递.客户端向服务器端发送"请输出从1到1000内所有 ...
- tomcat日志切割脚本
tomcat日志每俩小时切割的脚本如下(这是用定时任务来完成的,此方法无需重启tomcat): time=$(date +%H) end_time=`` a=$end_time BF_TIME=$(- ...
- JAVA / MySql 编程—— 第四章 高级查询(二)
1. EXISTS和NOT EXISTS子查询:EXISTS关键字用来检测数数据库对象是否存在. ★EXISTS和NOT EXISTS的结果只取决于是否 ...
- PHP递归操作
对于php的递归操作解释说明,递归基本上是学习每种语言都要会的最基本的操作.来吧,下面是我闲的时候随便写的一个对数组进行遍历操作的一个递归函数. 原理很简单,递归就是在一个函数里面调用自身的一种机制. ...
- get请求中url传参中文乱码问题
在项目中经常会遇到中文传参数,在后台接收到乱码问题.那么在遇到这种情况下我们应该怎么进行处理让我们传到后台接收到的参数不是乱码是我们想要接收的到的,下面就是我的一些认识和理解. 一:get请求url中 ...
- LayoutInflater.Factory 妙用
LayoutInflater.Factory 可以调试 布局,你把下面的代码放到 onCreate里面,然后再里面的下面的onCreateView里面加上断点,然后你就可以知道所有的view构造,所有 ...
- 利用NSAttributedString实现图文混排
UILabel 和 UITextView 都能添加 NSAttributedString 属性字符串,通过这一点,可以实现带有属性的文字和文字内包含图片的文本内容展示. 效果如下: 1-初始化可变 ...