这是CCPC女生专场的一道dp题。大佬们都说它简单,我并没有感到它有多简单。

先说一下题意:在一条直线上,有n个教室,现在我要在这些教室里从左到右地建设一些作为糖果屋,每个教室都有自己的坐标xi 和建造糖果屋的费用ci ,

如果在这里建造一个糖果屋,那么花费ci ,如果不建造糖果屋,则花费是当前教室的坐标与左边最靠近当前教室的糖果屋坐标之差,问最小花费。

一看这是个求最优解的问题,应该明白这是个dp问题,现在来考虑该问题状态的定义:

当我建设到第i个教室的时候,我有两种选择,建糖果屋或者不建糖果屋,影响我决策的是在左边离我最近的糖果屋的位置。并且我在建设前i个教室的时候,明显可以看出有i种相互独立的不同方案,即:

将离i最近的且在左边糖果屋建在  1号,2号,3号,......,k号,......,i号教室,所有的方案都可以分成这i类。我想知道建设前i个教室的最小费用,我只需挑出这i种方案中花费最小的那个就行。

并且,在我后面继续建设的时候,我一定还会用到这i种方案做基础,比如说我要建设第i+1个房间了,将最近的糖果屋建设在1~i已经有了现成的方案供我们选择,我们额外需要考虑的只是将糖果屋

建设在第i+1间教室时的情况。

于是我们定义状态dp[i][j]---->现在已经建设到第i个教室了,离i最近且在左边的的教室号为j时,我们能求得的最小花费。

状态转移方程:

if(i == j) dp[i][j] = minn[i - 1] + ci;

else dp[i][j] = dp[i - 1][j] + (xi - xj);

其中minn[k]指的是建设前k间教室的最小花费。

显然,当我们把最近的糖果屋建设在i号教室时,我们需要建设前i - 1间教室的最优解 + 建 i 号教室的费用,才能保证当前状态的最优解的生成。

这个minn[k]是可以顺带着算出来的,详见代码:

#include<stdio.h>
#include<string.h>
#include<iostream>
#include<algorithm>
#define maxn 3005
#define F 0x7f
#define INF 0x3f3f3f3f3f3f3f
using namespace std;
struct room
{
long long x,c;
};
room store[maxn];
long long dp[][maxn];
long long minn[maxn];
const long long zero = ;
bool cmp(const room& a,const room& b)
{
return a.x < b.x;
}
int main()
{
int n;
while(scanf("%d",&n) == ){
for(int i = ;i <= n;++i) dp[][i] = INF;
for(int i = ;i <= n;++i) dp[][i] = ;
for(int i = ;i <= n;++i) minn[i] = INF;
for(int i = ;i <= n;++i) scanf("%lld%lld",&store[i].x,&store[i].c);
sort(store + ,store + n + ,cmp);
dp[][] = store[].c;
minn[] = dp[][];
for(int i = ;i <= n;++i){
for(int j = ;j <= i;++j){
if(i == j) dp[i & ][j] = minn[i - ] + store[i].c;
else dp[i & ][j] = dp[(i - ) & ][j] + (store[i].x - store[j].x);
minn[i] = min(minn[i],dp[i & ][j]);
}
}
printf("%lld\n",minn[n]);
}
return ;
}

这里有一个特别坑的地方:标号小的教室坐标未必小,即这些教室不一定是从左到右标号的。

同时为了减少内存开销,我采用了滚动数组。

HDU 6024(中国大学生程序设计竞赛女生专场1002)的更多相关文章

  1. 2017中国大学生程序设计竞赛 - 女生专场 1002 dp

    Building Shops Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others) ...

  2. 2017中国大学生程序设计竞赛 - 女生专场(dp)

    Building Shops Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others) To ...

  3. 2017中国大学生程序设计竞赛 - 女生专场 Deleting Edges(思维+最短路)

    Deleting Edges Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others) ...

  4. 2017中国大学生程序设计竞赛 - 女生专场 Happy Necklace(递推+矩阵快速幂)

    Happy Necklace Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others) ...

  5. 2017中国大学生程序设计竞赛 - 女生专场(Graph Theory)

    Graph Theory Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others)To ...

  6. 2017中国大学生程序设计竞赛 - 女生专场B【DP】

    B HDU - 6024 [题意]:n个教室,选一些教室建造糖果商店. 每个教室,有一个坐标xi和在这个教室建造糖果商店的花费ci. 对于每一个教室,如果这个教室建造糖果商店,花费就是ci,否则就是与 ...

  7. hdu_5705_Clock("巴卡斯杯" 中国大学生程序设计竞赛 - 女生专场)

    题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=5705 题意:给你一个时间和一个角度,问你下一个时针和分针形成给出的角度是什么时候 题解:我们可以将这个 ...

  8. hdu_5707_Combine String("巴卡斯杯" 中国大学生程序设计竞赛 - 女生专场)

    题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=5707 题意:给你三个字符串 a,b,c,问你 c能否拆成a,b,a,b串的每一个字符在c中不能变 题解 ...

  9. 2017中国大学生程序设计竞赛 - 女生专场C【前后缀GCD】

    C HDU - 6025 [题意]:去除数列中的一个数字,使去除后的数列中所有数字的gcd尽可能大. [分析]: 数组prefixgcd[],对于prefixgcd[i]=g,g为a[0]-a[i]的 ...

随机推荐

  1. get_category_recommend_goods的正确使用

    get_category_recommend_goods($type = '', $cats = '', $brand = 0, $min =0,  $max = 0, $ext='') 位于lib_ ...

  2. mysql中 decimal、numeric数据类型

    例 如:salary DECIMAL(5,2) 在这个例子中,5 (精度(precision)) 代表重要的十进制数字的数目,2 (数据范围(scale)) 代表在小数点后的数字位数.在这种情况下,因 ...

  3. hive的安装和升级

    目录 [toc] 安装 升级 安装 nohup hive –service hiveserver & 启动: sudo nohup ./hive --service metastore &am ...

  4. Neuron:Neural activities in V1 create a bottom-up saliency map

    Neural activities in V1 create a bottom-up saliency map 本文证明了人类的初级视皮层可以在视觉信息加工的非常早期阶段,生成视觉显著图,用以引导空间 ...

  5. 初识bd时的一些技能小贴士

    既然小豆腐如此给力,而且充分的利用主动学习的优势,已经有了迅速脑补,压倒式的优势,不过这只是表面而已,一切才刚刚开始,究竟鹿死谁手,还有待验证. 以上可以看到,小豆腐为什么拼命的要teach我们了么, ...

  6. 关于/var/run/docker.sock

    译者按: 这篇博客介绍了什么是/var/run/docker.sock,以及如何使用/var/run/docker.sock与Docker守护进程通信,并且提供了两个简单的示例.理解这些,我们就可以运 ...

  7. Unity3D 正六边形,环状扩散,紧密分布,的程序

    最近在做一个正六边形的游戏,被一开始的布局难倒了. 需求:中心有个正六边形,输入围绕中心扩散的环数,自动创建和摆放. 大概就是这样的吧,我觉得这个非常轻松的就可以搞定了.啊~~~~~啊~~~ 五环~~ ...

  8. 《C++之那些年踩过的坑(附录一)》

    C++之那些年踩过的坑(附录一) 作者:刘俊延(Alinshans) 本系列文章针对我在写C++代码的过程中,尤其是做自己的项目时,踩过的各种坑.以此作为给自己的警惕. [版权声明]转载请注明原文来自 ...

  9. 关于Java中volatile关键字笔记

    volatile通常被认为是一种轻量级的synchronized,字面上它表示易变的,在并发编程中,它保证了共享变量的可见性.所谓可见性指的是,某个线程对变量进行操作后,其他线程能够读取到操作后的最新 ...

  10. .net之简单工厂模式

    //抽象一个生产笔的方法 namespace simpleFactory{ public abstract class AbstractPen { public abstract void creat ...