题目大意

给定 $N$ 个矩形的宽和高, $(h_1, w_1), (h_2, w_2), \dots, (h_n w_n)$ . 现需要确定 $k$ ($k \le K$, $K$ 给定) 个矩形 $(H_i, W_i)$ , 使得 $\forall i \ (1\le i\le N), \exists j,\ s.t.\ H_j \ge h_i$ 且 $W_j \ge w_i$, 此时称这后 $k$ 个矩形覆盖前 $N$ 个矩形, 代价为 $\sum_{1\le i \le k} H_i W_i$ , 即 $k$ 个矩形的面积之和. 求覆盖这 $N$ 个矩形的最小代价.

Solution

DP方程

对 $N$ 个矩形按 (宽, 高) 双关键字从小到大排序. 按此顺序构造一个高严格递减的矩形队列, 只考虑这个队列里的矩形, 记队列里的剩下的矩形数目为 $n, \ (n \le N)$ .

$dp[i][j]$ : 用 $j$ 个矩形覆盖队列中前 $i$ 个矩形的最小代价.

$$

\begin{equation}

dp[i][j] = \min {dp[i'][j-1]+w_i \times h_{i'+1}} \quad (0\le i' <i) \label{dp_eq}

\end{equation}

$$

边界条件: $dp[0][0]=0$

斜率优化

固定 $j$ , 用斜率优化 $i$ 这一维的决策.

关于斜率优化的细致讨论见 this post.

写出斜率不等式:

$$

\begin{equation}

g(i_1, i_2) < f(i) \quad (i_1<i_2) \label{general_ineq}

\end{equation}

$$

其中

$$ g(i_1, i_2) = \dfrac{dp[i_2][j-1] - dp[i_1][j-1]}{h[i_1+1]-h[i_2+1]}$$

$$ f(i) = w[i]$$

维护一个相邻点斜率严格递增的点的队列, 由于 $f(i)$ 单调递增 , 在维护单调队列的基础上, 计算 $dp[i][j]$ 之前, 若队首两点 $i_1, i_2$ 满足 $g(i_1, i_2) \le f(i)$ , 就将队首元素出队, 重复该过程. 将队首元素代入 $\ref{dp_eq}$ 式右边计算 $dp[i][j]$ 的值.

Implementation

#include <bits/stdc++.h>
using namespace std; const int N=5e4+5, K=105;
using ll=long long; ll dp[N][K]; // dp[i][j]=min()
int n, k;
ll w[N], h[N]; bool cmp(int x, int y){
return w[x]<w[y] || w[x]==w[y] && h[x]<h[y]; // error-prone
} int idx[N];
int q[N], t; // q[0]=0 void prep(){
for(int i=1; i<=n; i++)
idx[i]=i;
sort(idx+1, idx+n+1, cmp);
t=0;
// q[++tail]=idx[1];
for(int i=1; i<=n; i++){
while(t && h[q[t]]<=h[idx[i]])
--t;
q[++t]=idx[i];
}
} // x < y: index of q ll nu(int i, int x, int y){
return dp[y][i]-dp[x][i];
} ll de(int x, int y){
return h[q[x+1]]-h[q[y+1]];
} // the first dimension of dp[i][j] and q[i] are associated. int que[N], head, tail; int main(){
h[0]=1e6+5;
for(; cin>>n>>k; ){
for(int i=1; i<=n; i++)
scanf("%lld%lld", w+i, h+i); prep(); for(int j=0; j<=k; j++)
dp[0][j]=0;
for(int i=1; i<=t; i++)
dp[i][0]=1e12+5; //error-prone // p_i: (h[q[i+1]], dp[i])
for(int j=1; j<=k; j++){ // j: number of rectangles.
head=tail=0; // index of que
// 1. idx of q stored 2. 0-indexed
que[tail++]=0;
for(int i=1; i<=t; i++){ // i: index of q
while(tail-head>=2 &&
nu(j-1, que[head], que[head+1]) <= de(que[head], que[head+1])*w[q[i]])
++head; dp[i][j]=dp[que[head]][j-1]+h[q[que[head]+1]]*w[q[i]];
// printf("(%d,%d):%lld\n", i, j, dp[i][j]); while(tail-head>=2 &&
nu(j-1, que[tail-2], que[tail-1])*de(que[tail-1], i)
>=nu(j-1, que[tail-1], i)*de(que[tail-2], que[tail-1]))
--tail; que[tail++]=i; //error-prone
}
} ll res=LLONG_MAX;
for(int j=1; j<=k; j++)
res=min(res, dp[t][j]);
assert(res>0);
cout<<res<<endl;
}
return 0;
}

Pitfalls

如何将已经能被其他矩形覆盖的矩形剔除.

HDU 3669 Cross the Wall的更多相关文章

  1. hdu 3669 Cross the Wall(斜率优化DP)

    题目连接:hdu 3669 Cross the Wall 题意: 现在有一面无限大的墙,现在有n个人,每个人都能看成一个矩形,宽是w,高是h,现在这n个人要通过这面墙,现在只能让你挖k个洞,每个洞不能 ...

  2. 动态规划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 ...

  3. HDU 3669 Cross the Wall(斜率DP+预处理)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3669 题目大意:有n(n<=50000)个矩形,每个矩形都有高和宽,你可以在墙上最多挖k个洞使得 ...

  4. HDU 3669 [Cross the Wall] DP斜率优化

    问题分析 首先,如果一个人的\(w\)和\(h\)均小于另一个人,那么这个人显然可以被省略.如果我们将剩下的人按\(w[i]\)递增排序,那么\(h[i]\)就是递减. 之后我们考虑DP. 我们设\( ...

  5. hdu 3669(斜率优化DP)

    Cross the Wall Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 327680/327680 K (Java/Others) ...

  6. hdu 1543 Paint the Wall

    http://acm.hdu.edu.cn/showproblem.php?pid=1543 #include <cstdio> #include <cstring> #inc ...

  7. HDU 2124 Repair the Wall

    http://acm.hdu.edu.cn/showproblem.php?pid=2124 Problem Description Long time ago , Kitty lived in a ...

  8. HDU 4391 Paint The Wall(分块+延迟标记)

    Paint The Wall Time Limit: 20000/10000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) ...

  9. hdu3669 Cross the Wall

    Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 327680/327680 K (Java/Others) Total Submissi ...

随机推荐

  1. android SQLite数据库总结

    SQLite SQLite是一种超轻量级的嵌入式数据库,大小只有几百KB,但是其语法支持标准SQL语法,同时还遵循了数据库的ACID事务,所以学过其他数据库的开发人员都很容易掌握其使用. sql语法就 ...

  2. iOS - 静态库的创建与使用

    在日常项目开发中,不论是为了两个公司项目上的业务交流还是为了减少项目的编译时间,有的时候我们会把项目中的私密内容打包成静态库,或者是把项目中变动较少一部分打包成静态库以便提高编译效率,那么下面我们就来 ...

  3. android不需要Socket的跨进程推送消息AIDL!

    上篇介绍了跨进程实时通讯http://www.cnblogs.com/xiaoxiaing/p/5818161.html 但是他有个缺点就是服务端无法推送消息给客户端,今天这篇文章主要说的就是服务器推 ...

  4. java socket传送一个结构体给用C++编写的服务器解析的问题

    另一端是Java写客户端程序,两者之间需要通信.c++/c接收和发送的都是结构体,而Java是直接发送的字节流或者byte 数组.解决方法:c++/c socket 在发送结构体的时候其实发送的也是字 ...

  5. 6-3 bash脚本编程之五 字符串测试及for循环

    1. 字符测试 ==:等号两边要有空格,否则会被认为是赋值. !=:  测试是否相等,记住如果不等为真,等为假. -n string: 测试指定字符串是否为空,空位真,不空为假. -s string: ...

  6. 7、二种 为二个不同的子网配置DHCP服务器(中继代理服务器)

    环境如下:        (参考之前,保证二个子网可以互相ping通) 虚拟机vm1        192.168.170.3                    VMnet8 (NAT模式) 虚拟 ...

  7. python学习笔记-import utils报错

    今天遇到一个坑爹的问题,查找了半天原因,终于解决了,在此特地记录一下. 运行环境:Windows eclipse 我在eclipse中配置了python的运行环境,在eclipse中编写python代 ...

  8. Leetcode 57: Insert Interval 让代码更好读, 更容易测试.

    阅读了几个博客, 决定写一个简易版本; 忙着做更多题, 没有时间多考虑优化代码, 所以, 就写一个试试运气. http://blog.csdn.net/kenden23/article/details ...

  9. jsp九大内置对象

    application例如用于计算网站访问量时可用到.

  10. [No0000A8]Word中设置图片下的题注及插入多级列表编号

    1.什么是题注? 2.怎么实现一个可以自动更新的题注?  只有先定义好文档编号后,才可以设置出正确的图片下标题注. 文章的结构可以通过导航窗口导航. 导航窗口打开方式. 3.设置好文档编号后,怎样插入 ...