P1156 垃圾陷阱

4 篇文章 0 订阅

题目描述

卡门――农夫约翰极其珍视的一条Holsteins奶牛――已经落了到“垃圾井”中。“垃圾井”是农夫们扔垃圾的地方,它的深度为D(2 \le D \le 100)D(2≤D≤100)英尺。

卡门想把垃圾堆起来,等到堆得与井同样高时,她就能逃出井外了。另外,卡门可以通过吃一些垃圾来维持自己的生命。

每个垃圾都可以用来吃或堆放,并且堆放垃圾不用花费卡门的时间。

假设卡门预先知道了每个垃圾扔下的时间t(0< t \le 1000)t(0<t≤1000),以及每个垃圾堆放的高度h(1 \le h \le 25h(1≤h≤25)和吃进该垃圾能维持生命的时间f(1 \le f \le 30)f(1≤f≤30),要求出卡门最早能逃出井外的时间,假设卡门当前体内有足够持续1010小时的能量,如果卡门1010小时内没有进食,卡门就将饿死。

输入格式

第一行为22个整数,DD和 G (1 \le G \le 100)G(1≤G≤100),GG为被投入井的垃圾的数量。

第二到第G+1G+1行每行包括33个整数:T (0 < T <= 1000)T(0<T<=1000),表示垃圾被投进井中的时间;F (1 \le F \le 30)F(1≤F≤30),表示该垃圾能维持卡门生命的时间;和 H (1 \le H \le 25)H(1≤H≤25),该垃圾能垫高的高度。

输出格式

如果卡门可以爬出陷阱,输出一个整表示最早什么时候可以爬出;否则输出卡门最长可以存活多长时间。

输入输出样例

输入 #1复制

20 4
5 4 9
9 3 2
12 6 10
13 1 1

输出 #1复制

13

说明/提示

[样例说明]

卡门堆放她收到的第一个垃圾:height=9height=9;

卡门吃掉她收到的第22个垃圾,使她的生命从1010小时延伸到1313小时;

卡门堆放第33个垃圾,height=19height=19;

卡门堆放第44个垃圾,height=20height=20。

 思路:用f[i][j]来表示到第i堆的时候,高度到达j时剩下的最大生命值。

创建结构体数组p记录三个信息,即垃圾扔下的时刻t,吃了垃圾能维持的时间f,堆垃圾能增加的高度h。

因为有每堆垃圾要么堆要么吃,所以有两种状态转移方程:

用于堆(高度增加,剩下的最大生命值是第i-1堆到达高度j时的最大生命值减去第i堆和第i-1堆的时间差):

f[i] [j+p[i].h]=max(f[i] [j+h], f[i-1] [j]- (p[i].t-p[i-1].t) )

用于吃(高度不变,剩下的最大生命值是第i-1堆到达j高度时剩下的最大生命值减去第i堆和第i-1堆的时间差+吃了第i堆能维持的时间)

f[i] [j]=max(f[i] [j],f[i-1][j]-(p[i].t-p[i-1].t)+p[i].f)

只有当我们第i-1堆剩余的最大生命值大于等于从第i-1堆到第i堆相差的时间我们才能进行第i堆的转换,所以当进行上面两个转移之前我们要先判断一下是否符合条件。

那么能逃出去的时刻,就是能进行状态转移,并且到达的高度j加上第i堆的h大于等于d的时刻,因为堆垃圾不费时间,所以直接就是扔下第i堆的时刻。直接判断如果满足条件直接输出i的时刻即可。

那么如果不能逃出去,就是我们把能吃的垃圾全都吃掉所维持的时间。ans表示最大能维持的时刻,刚开始为10,ans=10。然后遍历一遍p数组,如果最大能维持的时刻小于第i堆垃圾扔下来的时刻,那么说明在第i堆扔下之前就死了,所以我们直接break就行,否则的话说明在第i堆扔下来的时刻没有死,那么就加上吃掉第i堆维持的时间f。最后输出ans即可。

#include<iostream>
#include<algorithm>
#include<cstring>

using namespace std;
int d,n;
const int N=105;
struct name{
	int t,f,h;
};
name p[N];
int f[N][N];
bool cmp(name a,name b){
	return a.t <b.t ;
}
int main(){
	cin>>d>>n;
	for(int i=1;i<=n;i++){
		cin>>p[i].t >>p[i].f>>p[i].h ;
	}
	sort(p+1,p+1+n,cmp);
	memset(f,-1,sizeof f);
	f[0][0]=10;
	p[0].t =0;
	for(int i=1;i<=n;i++){
		for(int j=d;j>=0;j--){//相当于用01背包,滚动数组
			if(f[i-1][j]>=p[i].t -p[i-1].t ){//如果他剩下的最大时间大于到第i堆需要的时间
				if(j+p[i].h >=d){
					cout<<p[i].t ;
					return 0;
				}
				f[i][j+p[i].h ]=max(f[i][j+p[i].h ],f[i-1][j]-p[i].t +p[i-1].t );
				f[i][j]=max(f[i][j],f[i-1][j]-p[i].t +p[i-1].t +p[i].f );
			}
		}
	}
	int ans=10;//最大活的时间
	for(int i=1;i<=n;i++){
		if(ans<p[i].t )break;
		ans+=p[i].f ;
	}
	cout<<ans;
	return 0;
}

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值