POJ2430 Lazy Cows
题意描述
给定一个 \(2\times b\) 的矩形,和 \(n\) 个矩形上的点。
要求你用 \(k\) 个矩形覆盖这 \(n\) 个点,使得每个点都被覆盖的前提下这些矩形的面积和最小。
算法分析
这道题的阶段性很强(按照目标点的纵坐标),但是状态不太好表示,于是想到状压。
首先将图改变一下,便于 DP:
- 将输入的奶牛按照位置从小到大排序。
- 在每个不同的位置(横坐标)记录一次。
- 如果这个位置(横坐标)的仅上面有奶牛,标记为 \(1\);仅下面有奶牛,标记为 \(2\);上下都有,标记为 \(3\)。
那么显然,改变之后的图中不同牛的数量 \(\leq n\)。
设计状态
设 \(f(i,j,0\)~\(4)\) 表示:到第 \(i\) 个牛的位置,用了 \(j\) 个牛舍,状态为 \(0\)~\(4\) 的最小面积。
解释一下状态:
- 表示上下都没有牛舍,这种情况仅存在于初始化。
- 表示上面有牛舍,下面没有。
- 表示上面没有牛舍,下面有。
- 表示上下都有牛舍,而且是同一个牛舍。
- 表示上下都有牛舍,而且是不同牛舍。
预处理
就是...,酱紫:
\(f(0,0,0)=0\)
\(f(1,1,1)=f(1,1,2)=1\)
\(f(1,1,3)=f(1,2,4)=2\)
\(f(其他)=INF\)
状态转移方程
然后我们按照情况讨论(推柿子)即可:
情况一:不增加牛舍数量
设 \(tmp=cow[i].x-cow[i-1].x\)(前后两列奶牛的横坐标之差)
\(f(i,j,1)=min(f(i-1,j,1),f(i-1,j,4))+tmp\)
\(f(i,j,2)=min(f(i-1,j,2),f(i-1,j,4))+tmp\)
\(f(i,j,3)=f(i-1,j,3)+2\times tmp\)
\(f(i,j,4)=f(i-1,j,4)+2\times tmp\)
解释一下:当不新增牛舍时,只能延长原本存在的牛舍,易得上面的递推式。
情况二:增加一个牛舍
设:
- \(best1=min(f(i-1,j-1,1),f(i-1,j-1,2))\)
- \(best2=min(f(i-1,j-1,3),f(i-1,j-1,4))\)
- \(best=min(best1,best2)\)
\(f(i,j,1)=min(f(i,j,1),best+1)\)
\(f(i,j,2)=min(f(i,j,2),best+1)\)
\(f(i,j,3)=min(f(i,j,3),best+2)\)
\(f(i,j,4)=min(f(i,j,4),min(best1,f(i-1,j-1,4))+(tmp+1))\)
再来解释一下:首先的预处理就是为了方便处理,仅是个人习惯不用过多纠结。
如果能够增加一个牛舍,那么对于状况 \(1,2,3\) 均可以直接原地增加一个牛舍,不用管前面是什么状况。
但是对于状况 \(4\),只能增加一个牛舍的情况将十分尴尬,只能从前面延长一个牛舍,再本地新增一个牛舍。
显然只能从上一次的状况 \(1,2,4\) 推来。
情况三:增加两个牛舍
\(f(i,j,4)=min(f(i,j,4),min\{f(i,j-2,1\)~\(4)\}+2)\)
显然,只有情况四需要新增两个牛舍(其他一个就够了),所以易得上方程。
答案统计
易得:
\(ans=min_{1\leq p\leq 4}\{f(n,k,p)\}\)
代码实现
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<iostream>
#include<cmath>
#define N 1010
#define INF 0x3f3f3f3f
using namespace std;
int n,k,b,cnt=0;
int f[N][N][10];
struct node{
int x,y;
}a[N];
struct Cow{
int x,t;
}cow[N];
int read(){
int x=0,f=1;char c=getchar();
while(c<'0' || c>'9') f=(c=='-')?-1:1,c=getchar();
while(c>='0' && c<='9') x=x*10+c-48,c=getchar();
return x*f;
}
bool cmp(node a,node b){
if(a.x!=b.x) return a.x<b.x;
return a.y<b.y;
}
void build(){
for(int i=1;i<=n;i++){
if(a[i].x==a[i-1].x)
cow[cnt].t=3;
else cow[++cnt].t=a[i].y,cow[cnt].x=a[i].x;
}
return;
}
void dp(){
memset(f,0x3f,sizeof(f));
f[0][0][0]=0;
if(cow[1].t==1) f[1][1][1]=1;
else if(cow[1].t==2) f[1][1][2]=1;
f[1][1][3]=f[1][2][4]=2;
for(int i=2;i<=cnt;i++){
for(int j=1;j<=k;j++){
int tmp=cow[i].x-cow[i-1].x;
if(cow[i].t==1) f[i][j][1]=min(f[i-1][j][1],f[i-1][j][4])+tmp;
if(cow[i].t==2) f[i][j][2]=min(f[i-1][j][2],f[i-1][j][4])+tmp;
f[i][j][3]=f[i-1][j][3]+2*tmp;
f[i][j][4]=f[i-1][j][4]+2*tmp;
if(j==1) continue;
int best1=min(f[i-1][j-1][1],f[i-1][j-1][2]);
int best2=min(f[i-1][j-1][3],f[i-1][j-1][4]);
int best=min(best1,best2);
if(cow[i].t==1) f[i][j][1]=min(f[i][j][1],best+1);
if(cow[i].t==2) f[i][j][2]=min(f[i][j][2],best+1);
f[i][j][3]=min(f[i][j][3],best+2);
f[i][j][4]=min(f[i][j][4],min(f[i-1][j-1][4],best1)+(tmp+1));
if(j==2) continue;
f[i][j][4]=min(f[i][j][4],min(min(f[i-1][j-2][1],f[i-1][j-2][2]),min(f[i-1][j-2][3],f[i-1][j-2][4]))+2);
}
}
}
int main(){
//freopen("lazy.in","r",stdin);
//freopen("lazy.out","w",stdout);
n=read(),k=read(),b=read();
for(int i=1;i<=n;i++)
a[i].y=read(),a[i].x=read();
sort(a+1,a+n+1,cmp);
build();
dp();
printf("%d\n",min(min(f[cnt][k][1],f[cnt][k][2]),min(f[cnt][k][3],f[cnt][k][4])));
//fclose(stdin);fclose(stdout);
return 0;
}
完结撒花
POJ2430 Lazy Cows的更多相关文章
- DP百题练(二)
目录 DP百题练(二) 区间 DP NOI1995 石子合并 IOI1998 Polygon CH5302 金字塔 USACO06FEB Treats for the Cows G/S LG1043 ...
- 别人整理的DP大全(转)
动态规划 动态规划 容易: , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , ...
- dp题目列表
此文转载别人,希望自己能够做完这些题目! 1.POJ动态规划题目列表 容易:1018, 1050, 1083, 1088, 1125, 1143, 1157, 1163, 1178, 1179, 11 ...
- 杭电ACM分类
杭电ACM分类: 1001 整数求和 水题1002 C语言实验题——两个数比较 水题1003 1.2.3.4.5... 简单题1004 渊子赛马 排序+贪心的方法归并1005 Hero In Maze ...
- poj 动态规划题目列表及总结
此文转载别人,希望自己能够做完这些题目! 1.POJ动态规划题目列表 容易:1018, 1050, 1083, 1088, 1125, 1143, 1157, 1163, 1178, 1179, 11 ...
- [转] POJ DP问题
列表一:经典题目题号:容易: 1018, 1050, 1083, 1088, 1125, 1143, 1157, 1163, 1178, 1179, 1189, 1191,1208, 1276, 13 ...
- poj动态规划列表
[1]POJ 动态规划题目列表 容易: 1018, 1050, 1083, 1088, 1125, 1143, 1157, 1163, 1178, 1179, 1189, 1208, 1276, 13 ...
- POJ动态规划题目列表
列表一:经典题目题号:容易: 1018, 1050, 1083, 1088, 1125, 1143, 1157, 1163, 1178, 1179, 1189, 1191,1208, 1276, 13 ...
- POJ 动态规划题目列表
]POJ 动态规划题目列表 容易: 1018, 1050, 1083, 1088, 1125, 1143, 1157, 1163, 1178, 1179, 1189, 1208, 1276, 1322 ...
随机推荐
- 介绍使用Cordova和Web Starter Kit开发Android
介绍 如今,每个人都想制作移动应用程序,为什么不呢?世界上有更多的移动设备比任何其他用户设备.Android尤其流行,但是为什么不从一个众所周知的跨平台应用的基础开始呢?Android的开发显然比其他 ...
- 对ACE和ATL积分
下载source code - 39.66 KB 介绍 这篇文章展示了一种结合ACE和ATL的方法.它不打算作为功能演示,而是作为一个小型的"入门"解决方案,展示实现此目标的可行方 ...
- CPU 执行程序的秘密,藏在了这 15 张图里
前言 代码写了那么多,你知道 a = 1 + 2 这条代码是怎么被 CPU 执行的吗? 软件用了那么多,你知道软件的 32 位和 64 位之间的区别吗?再来 32 位的操作系统可以运行在 64 位的电 ...
- 二进制部署Redis-5.07
Redis简介 Redis是一个开源(BSD许可),内存存储的数据结构服务器,可用作数据库,高速缓存和消息队列代理. 它支持字符串.哈希表.列表.集合.有序集合,位图,hyperloglogs等数据类 ...
- 使用docker搭建redis服务器记录
#mkdir /home/redishome#mkdir /home/redishome/data#chmod -R 777 /home/redishome把redis.conf传到/home/red ...
- ZooKeeper伪分布式集群安装及使用
ZooKeeper伪分布式集群安装及使用 让Hadoop跑在云端系列文章,介绍了如何整合虚拟化和Hadoop,让Hadoop集群跑在VPS虚拟主机上,通过云向用户提供存储和计算的服务. 现在硬件越来越 ...
- 多测师讲解 自动化测试理论(1)_高级讲师肖sir
自动化测试理论什么是自动化测试?广义的:通过工具或程序替代或辅助人工测试的行为叫自动化测试狭义的:通过工具录制或编写脚本模拟手工测试的过程,通过回放或运行脚本执行测试用例,从而代替人工对系统的功能验证 ...
- 互不侵犯(洛谷P1896)
题目:在N*N的棋盘里面放k个国王,使他们互不攻击,共有多少种摆放方案.国王能攻击到它上下左右,以及左上左下右上右下八个方向上附近的各一个格子,共8个格子. 输入输出:输入N,K,输出有几种放置方法. ...
- kali 運行 chrome
0x00前提 已經安裝好google chrome . 0x01 在終端執行命令: google-chrome,發現如圖: 錯誤提示:在root下只能使用 --no-sandbox選項來運行chrom ...
- JetBrains-CLion永久激活教程
前言 JetBrains Clion 是一款专为 C/C++ 开发所设计的跨平台 IDE.本文适用 JetBrains CLion v2019.3/3.1/3.2/3.3 永久激活,附破解补丁和激活码 ...