题意描述

Lazy Cows

给定一个 \(2\times b\) 的矩形,和 \(n\) 个矩形上的点。

要求你用 \(k\) 个矩形覆盖这 \(n\) 个点,使得每个点都被覆盖的前提下这些矩形的面积和最小。

算法分析

这道题的阶段性很强(按照目标点的纵坐标),但是状态不太好表示,于是想到状压。

首先将图改变一下,便于 DP:

  1. 将输入的奶牛按照位置从小到大排序。
  2. 在每个不同的位置(横坐标)记录一次。
  3. 如果这个位置(横坐标)的仅上面有奶牛,标记为 \(1\);仅下面有奶牛,标记为 \(2\);上下都有,标记为 \(3\)。

那么显然,改变之后的图中不同牛的数量 \(\leq n\)。

设计状态

设 \(f(i,j,0\)~\(4)\) 表示:到第 \(i\) 个牛的位置,用了 \(j\) 个牛舍,状态为 \(0\)~\(4\) 的最小面积。

解释一下状态:

  1. 表示上下都没有牛舍,这种情况仅存在于初始化。
  2. 表示上面有牛舍,下面没有。
  3. 表示上面没有牛舍,下面有。
  4. 表示上下都有牛舍,而且是同一个牛舍。
  5. 表示上下都有牛舍,而且是不同牛舍。

预处理

就是...,酱紫:

\(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\)

解释一下:当不新增牛舍时,只能延长原本存在的牛舍,易得上面的递推式。

情况二:增加一个牛舍

设:

  1. \(best1=min(f(i-1,j-1,1),f(i-1,j-1,2))\)
  2. \(best2=min(f(i-1,j-1,3),f(i-1,j-1,4))\)
  3. \(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的更多相关文章

  1. DP百题练(二)

    目录 DP百题练(二) 区间 DP NOI1995 石子合并 IOI1998 Polygon CH5302 金字塔 USACO06FEB Treats for the Cows G/S LG1043 ...

  2. 别人整理的DP大全(转)

    动态规划 动态规划 容易: , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , ...

  3. dp题目列表

    此文转载别人,希望自己能够做完这些题目! 1.POJ动态规划题目列表 容易:1018, 1050, 1083, 1088, 1125, 1143, 1157, 1163, 1178, 1179, 11 ...

  4. 杭电ACM分类

    杭电ACM分类: 1001 整数求和 水题1002 C语言实验题——两个数比较 水题1003 1.2.3.4.5... 简单题1004 渊子赛马 排序+贪心的方法归并1005 Hero In Maze ...

  5. poj 动态规划题目列表及总结

    此文转载别人,希望自己能够做完这些题目! 1.POJ动态规划题目列表 容易:1018, 1050, 1083, 1088, 1125, 1143, 1157, 1163, 1178, 1179, 11 ...

  6. [转] POJ DP问题

    列表一:经典题目题号:容易: 1018, 1050, 1083, 1088, 1125, 1143, 1157, 1163, 1178, 1179, 1189, 1191,1208, 1276, 13 ...

  7. poj动态规划列表

    [1]POJ 动态规划题目列表 容易: 1018, 1050, 1083, 1088, 1125, 1143, 1157, 1163, 1178, 1179, 1189, 1208, 1276, 13 ...

  8. POJ动态规划题目列表

    列表一:经典题目题号:容易: 1018, 1050, 1083, 1088, 1125, 1143, 1157, 1163, 1178, 1179, 1189, 1191,1208, 1276, 13 ...

  9. POJ 动态规划题目列表

    ]POJ 动态规划题目列表 容易: 1018, 1050, 1083, 1088, 1125, 1143, 1157, 1163, 1178, 1179, 1189, 1208, 1276, 1322 ...

随机推荐

  1. 小白也能看懂的Redis教学基础篇——朋友面试被Skiplist跳跃表拦住了

    各位看官大大们,双节快乐 !!! 这是本系列博客的第二篇,主要讲的是Redis基础数据结构中ZSet(有序集合)底层实现之一的Skiplist跳跃表. 不知道那些是Redis基础数据结构的看官们,可以 ...

  2. 067 01 Android 零基础入门 01 Java基础语法 08 Java方法 05 数组作为方法参数

    067 01 Android 零基础入门 01 Java基础语法 08 Java方法 05 数组作为方法参数 本文知识点:数组作为方法参数 说明:因为时间紧张,本人写博客过程中只是对知识点的关键步骤进 ...

  3. 如何查找一个为NULL的MYSQL字段

    前言:在做这个题目 https://www.cnblogs.com/pipihao/p/13786304.html 因为之前 我好像没有接触过什么 为NULL字段的查询,细节不小 WHERE 字段 I ...

  4. ubuntu19.10 系统需要安装的软件

    将ubuntu18 升级到ubuntu19 期间好几次卡在启动界面,比较担心要不要重装系统,有幸后来正常了.明显感觉操作快了不少.下半年稳定版就出来,到时候免不了再折腾一番,提前把安全记录做好. 下面 ...

  5. 初始python的类

    面向对象 一.面向对象 优点: 面向对象编程:是一类相似功能函数的集合,使你的代码更清晰化,更合理化. 面向对象,要拥有上帝的视角看问题,类其实就是一个公共模板,对象就从具体的模板实例化出来 类:就是 ...

  6. 基于Python的接口自动化实战-基础篇之读写配置文件

    引言 在编写接口自动化测试脚本时,有时我们需要在代码中定义变量并给变量固定的赋值.为了统一管理和操作这些固定的变量,咱们一般会将这些固定的变量以一定规则配置到指定的配置文件中,后续需要用到这些变量和变 ...

  7. 用-pthread替代-lpthread

    -pthread 在多数系统中,-pthread会被展开为"-D_REENTRANT -lpthread".作为编译参数可以通知系统函数开启多线程安全特性,比如将errno定义线程 ...

  8. 我是先学C语言还是先学C++,实不相瞒,鱼和熊掌可兼得!

    这是最近一周时间几个读者小伙伴所提的问题,我顺手截了两个图. 实不相瞒,这类问题之前也经常看到.     每次遇到这种问题,看起来很简单,但是打字一时半会还真说不清,想想今天周末了,写一篇文章来统一聊 ...

  9. ansible用authorized_key模块批量推送密钥到受控主机(免密登录)(ansible2.9.5)

    一,ansible的authorized_key模块的用途 用来配置密钥实现免密登录: ansible所在的主控机生成密钥后,如何把公钥上传到受控端? 当然可以用ssh-copy-id命令逐台手动处理 ...

  10. c++数组的替代品