喜欢玩warcraft的ltl

喜欢玩warcraft的ltl

时间限制: 2000 ms  |  内存限制: 65535 KB
难度: 3
描述

ltl 非常喜欢玩warcraft,因为warcraft十分讲究团队整体实力,而他自己现在也为升级而不拖累团队而努力。

他现在有很多个地点来选择去刷怪升级,但是在每一个地点他都要买上充足的补给和合适的道具,以免在刷怪的时候被怪物反杀了,每一个地点的怪物打完了就没有了(还居然不掉金钱跟装备),而且他只要选定了地点就一定会刷完该地点全部的怪物,同时获得对应的经验值。现在ltl 能给出每一个地点用来买补给和道具的钱和打完全部怪物所能获得的经验,但是他所拥有的钱是一定的。所以他想知道怎么选择地点使得他获得的经验最高。

输入
第一行一个整数T,表示测试数据的组数 0<T<=10
第二行两个整数N,M,0<N<=100,0<M<=1000000表示ltl拥有N个不同地点的选择和M的金钱总数
接下来N行每行两个整数ci,vi,(0<ci<=1000000,0<vi<=2000)表示ltl刷完第i个地点所需要购买补给和道具的总钱数和能获取的总经验值
输出
一行一个整数,表示ltl能够获取的最大的经验值
样例输入
2
3 10
7 7
2 3
3 5
2 5
3 5
2 1
样例输出
Max experience: 12
Max experience: 6
读完题,立马断定01背包问题,然后直接写代码,不幸的是,TLE不期而至!
超时代码:
#include<stdio.h>
struct node
{
	int c;
	int w;
}num[105];
int dp[1000005];
int Max(int a,int b)
{
	return a>b?a:b;
}
int main()
{
	int T,n,m;
	int i,j;
	scanf("%d",&T);
	while(T--)
	{
		scanf("%d%d",&n,&m);
		for(i=0;i<n;i++)
		{
			scanf("%d%d",&num[i].c,&num[i].w);
		}
		for(i=0;i<n;i++)
		{
			for(j=m;j>=num[i].c;j--)
			{
				dp[j]=Max(dp[j],dp[j-num[i].c]+num[i].w);
			}
		}
		printf("Max experience: %d\n",dp[m]);
	}
	return 0;
}

以下为优化代码:
#include<stdio.h>
struct node
{
	int c;
	int w;
}num[105];
int dp[1000005];
int Max(int a,int b)
{
	return a>b?a:b;
}
int main()
{
	int T,n,m;
	int i,j,s,count;
	scanf("%d",&T);
	while(T--)
	{
		scanf("%d%d",&n,&m);
		s=0;
		for(i=0;i<n;i++)
		{
			scanf("%d%d",&num[i].c,&num[i].w);
			s+=num[i].c;
		}
		for(i=0;i<=m;i++)
			dp[i]=0;
		for(i=0;i<n;i++)
		{
			s-=num[i].c;
			count=Max(m-s,num[i].c);
			for(j=m;j>=count;j--)
			{
				dp[j]=Max(dp[j],dp[j-num[i].c]+num[i].w);
			}
		}
		printf("Max experience: %d\n",dp[m]);
	}
	return 0;
}

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

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值