描述

佳佳最近又迷上了某款类似于虚拟人生的网络游戏。在游戏中,佳佳是某旅行团的团长,他需要安排客户住进旅馆。旅馆给了佳佳的旅行团一个房间数的限制。每一个房间有不同的容纳人数和价钱(这个价格是房间的总价格,不是每个人付的)。佳佳决定找到最小的花费,安排参加旅行的人住在这里。但是他遇到了这么一个问题:两个不同性别的人不能住在同一个房间里,除非他们是夫妻;一对夫妻如果在一起住,那么别的人就不能再住进去。你不必让所有的夫妻都单独住在一起。也就是说:
1.给你一些房间,告诉你这些房间的容纳人数和价格
2.安排一定数量的人住到旅馆里,满足:
a.不同性别的人如果不是夫妻那么不能住一起。
b.夫妻如果住在一起,那么房间不能安排其他的人进去。

你来写一个程序帮助佳佳找到安排这些来参加旅行的人住进旅馆所需要的最小花费。

格式

输入格式

第一行有4个用空格隔开的整数m,f,r,c,分别表示参加旅行的男性人数、参加旅行的女性人数、旅馆的房间数、这些男女中有多少对夫妻。注意每一个人不是单身就是和他/她唯一的妻子/丈夫一起参加旅行。

接下来有r行,每行描述了一个房间。每行有两个整数Bi,Pi,它们分别表示每一个房子的容纳人数和价格(无论住多少人,房间的价格不变)。

对于30%的数据,0<=m,f,r<=50;
对于100%的数据,0<=m,f,r<=300,0<=c<=Min(m,f),0<=Bi,Pi<=10。

输出格式

输出为旅行的人订购房间所需要的最小花费。如果没有这样的安排,请输出“Impossible”代替。

样例1

样例输入1[复制]

2 1 3 1
3 5
2 10
2 4

样例输出1[复制]

 
9

限制

各测试点5秒

-----------------------------------------

总存在一个最优解使得只有一对couple在一起或者没有,因为假如有两队,可以男女交换

f[i][j][k]和g[i][j][k]分别表示没有和有一对,转移时除了f向f,g向g外,再考虑g[i][j][k]=min(g[i][j][k],f[i-1][j-1][k-1]+p[i])

【WARN】:1.注意这个转移g[i][j][k]=min(g[i][j][k],f[i-1][j-1][k-1]+p[i]) j和k>=1

2.房间不一定注满人,如果再加一层枚举房间住几个人会很慢,可以把j和k<0的情况从不合法改成从0转移,想当于考虑了房间不满,最多男女各有一个房间不满

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int N=,INF=1e9;
int m,fm,r,c,b[N],p[N];
int f[N][N][N],g[N][N][N]; void dp(){
for(int j=;j<=m;j++) for(int k=;k<=fm;k++) f[][j][k]=g[][j][k]=INF;
f[][][]=;
for(int i=;i<=r;i++)
for(int j=;j<=m;j++)
for(int k=;k<=fm;k++){
f[i][j][k]=f[i-][j][k]; g[i][j][k]=g[i-][j][k];
if(b[i]>=&&j>=&&k>=) g[i][j][k]=min(g[i][j][k],f[i-][j-][k-]+p[i]);
int sj=max(,j-b[i]),sk=max(,k-b[i]);
f[i][j][k]=min(f[i][j][k],f[i-][sj][k]+p[i]),
g[i][j][k]=min(g[i][j][k],g[i-][sj][k]+p[i]);
f[i][j][k]=min(f[i][j][k],f[i-][j][sk]+p[i]),
g[i][j][k]=min(g[i][j][k],g[i-][j][sk]+p[i]);
}
}
int main(){
scanf("%d%d%d%d",&m,&fm,&r,&c);
for(int i=;i<=r;i++) scanf("%d%d",&b[i],&p[i]);
dp(); int ans;
if(c==) ans=f[r][m][fm];
else ans=min(f[r][m][fm],g[r][m][fm]);
if(ans!=INF) cout<<ans;
else cout<<"Impossible";
}
//题解中神犇的滚动数组做法 边读边做
//快了近三倍
#include <iostream>
#include <cstdio>
#include <cmath>
#include <cstring>
#include <algorithm>
#include <queue>
using namespace std;
typedef long long LL;
const int N = + ;
int m,f,r,c;
int dp[N][N][];
int main()
{
scanf("%d%d%d%d",&m,&f,&r,&c); memset(dp,0x3f3f3f3f,sizeof(dp));
dp[m][f][] = dp[m][f][] = ;
for(int i = ; i < r; i++)
{
int a,b;
scanf("%d%d",&a,&b);
for(int i = ; i <= m; i++)
{
for(int j = ; j <= f; j++)
{
int x1 = max(,i-a);
int x2 = max(,j-a);
dp[x1][j][] = min(dp[x1][j][],dp[i][j][] + b);
dp[i][x2][] = min(dp[i][x2][],dp[i][j][] + b);
dp[x1][j][] = min(dp[x1][j][],dp[i][j][] + b);
dp[i][x2][] = min(dp[i][x2][],dp[i][j][] + b);
if(i > && j > && a >= )
dp[i-][j-][] = min(dp[i-][j-][],dp[i][j][] + b);
}
}
}
int d = min(dp[][][],dp[][][]);
if(d == 0x3f3f3f3f)
printf("Impossible");
else
printf("%d",d); }

VIJOS1240 朴素的网络游戏[DP]的更多相关文章

  1. 洛谷 P3343 - [ZJOI2015]地震后的幻想乡(朴素状压 DP/状压 DP+微积分)

    题面传送门 鸽子 tzc 竟然来补题解了,奇迹奇迹( 神仙题 %%%%%%%%%%%% 解法 1: 首先一件很明显的事情是这个最小值可以通过类似 Kruskal 求最小生成树的方法求得.我们将所有边按 ...

  2. 清北学堂2018DP&图论精讲班 DP部分学习笔记

    Day 1 上午 讲的挺基础的--不过还是有些地方不太明白 例1 给定一个数n,求将n划分成若干个正整数的方案数. 例2 数字三角形 例7 最长不下降子序列 以上太过于基础,不做深入讨论 例3 给定一 ...

  3. SGU_390_Tickets(另类数位DP)

    Tickets Time Limit : 1000/500ms (Java/Other)   Memory Limit : 524288/262144K (Java/Other) Total Subm ...

  4. 斜率dp+cdq分治

    写在前面 这个东西应该是一个非常重要的套路......所以我觉得必须写点什么记录一下,免得自己忘掉了 一直以来我的斜率dp都掌握的不算很好......也很少主动地在比赛里想到 写这个的契机是noi.a ...

  5. [51nod 1022] 石子归并v2 [dp+四边形不等式优化]

    题面: 传送门 思路: 加强版的石子归并,现在朴素的区间dp无法解决问题了 首先我们破环成链,复制一条一样的链并粘贴到原来的链后面,变成一个2n长度的序列,在它上面dp,效率O(8n^3) 显然是过不 ...

  6. 【BZOJ2286】消耗战(虚树,DFS序,树形DP)

    题意:一棵N个点的树上有若干个关键点,每条边有一个边权,现在要将这些关键点到1的路径全部切断,切断一条边的代价就是边权. 共有M组询问,每组询问有k[i]个关键点,对于每组询问求出完成任务的最小代价. ...

  7. 「USACO16OPEN」「LuoguP3146」248(区间dp

    题目描述 Bessie likes downloading games to play on her cell phone, even though she doesfind the small to ...

  8. Luogu P2858 [USACO06FEB]奶牛零食Treats for the Cows 【区间dp】By cellur925

    题目传送门 做完A Game以后找道区间dp练练手...结果这题没写出来(哭). 和A Game一样的性质,从两边取,但是竟然还有天数,鉴于之前做dp经常在状态中少保存一些东西,所以这次精心设计了状态 ...

  9. 区间DP的瞎扯淡

    写在前面连个引言都不加就直接开1. 区间DP状态常见模板: f[i][j]常常表示第i个到第j个这个区间内达到题目要求,所需要的最小值(最大值) 如: 1. [石子合并](https://www.lu ...

随机推荐

  1. [deviceone开发]-课程表的例子

    一.简介 这个例子是根据一个真实app的一个页面的需求来实现的demo,通过动态add ui的方式,动态bind数据构建一个完整的课程表示例.示例并不完善,但是可以给大家一个启发. 二.效果图 三.相 ...

  2. Javascript数组算法和技巧总结

    Js-arrayMethod https://github.com/HerbertKarajan/Js-arrayMethod List unique an array 数组去重 random str ...

  3. 百度在线编辑器UEditor(v1.3.6) .net环境下详细配置教程之更改图片和附件上传路径

    本文是接上一篇博客,如果有疑问请先阅读上一篇:百度在线编辑器UEditor(v1.3.6) .net环境下详细配置教程 默认UEditor上传图片的路径是,编辑器包目录里面的net目录下 下面就演示如 ...

  4. 直接拿来用!十大Material Design开源项目

    来自:http://www.csdn.net/article/2014-11-21/2822753-material-design-libs/1 介于拟物和扁平之间的Material Design自面 ...

  5. Hbase Java API详解

    HBase是Hadoop的数据库,能够对大数据提供随机.实时读写访问.他是开源的,分布式的,多版本的,面向列的,存储模型. 在讲解的时候我首先给大家讲解一下HBase的整体结构,如下图: HBase ...

  6. Android自定义ScrollView分段加载大文本数据到TextView

    以下内容为原创,转载时请注明链接地址:http://www.cnblogs.com/tiantianbyconan/p/3311658.html 这是我现在碰到的一个问题,如果需要在TextView中 ...

  7. App 即时通讯 SDK

    1.网易云信 http://netease.im/ 2.环信 http://www.easemob.com/customer/im 3.融云 http://www.rongcloud.cn/ 4.极光 ...

  8. Android Material design

    1.Material Design:扁而不平 2.Android Support Design 库 之 Snackbar使用及源码分析 3.十大Material Design开源项目,直接拿来用!

  9. Ubuntu16.04 LTS 下安装 Android Studio 2.2.2 步骤

    下载 Android SDK,官网:www.android.com 我相信很多人跟我一样,进不去.Android Studio 中文官网 www.android-studio.org 进去下载, 在终 ...

  10. 高仿精仿手机版QQ空间应用源码

    说明:本次QQ空间更新了以前非常基础的代码 更新内容一 更新了登陆界面二  增加了输入时密码时和登陆成功后播放音频的效果三 增加了导航条渐隐的效果(和真实QQ空间的导航条一样,首先透明,当tablev ...