FIRST
——————————————————————
1102 采药 2005年NOIP全国联赛普及组
时间限制: 1 s
空间限制: 128000 KB
题目等级 : 黄金 Gold
题解
题目描述 Description
辰辰是个天资聪颖的孩子,他的梦想是成为世界上最伟大的医师。为此,他想拜附近最有威望的医师为师。医师为了判断他的资质,给他出了一个难题。医师把他带到一个到处都是草药的山洞里对他说:“孩子,这个山洞里有一些不同的草药,采每一株都需要一些时间,每一株也有它自身的价值。我会给你一段时间,在这段时间里,你可以采到一些草药。如果你是一个聪明的孩子,你应该可以让采到的草药的总价值最大。”
如果你是辰辰,你能完成这个任务吗?
输入描述 Input Description
输入第一行有两个整数T(1<=T<=1000)和M(1<=M<=100),用一个空格隔开,T代表总共能够用来采药的时间,M代表山洞里的草药的数目。接下来的M行每行包括两个在1到100之间(包括1和100)的整数,分别表示采摘某株草药的时间和这株草药的价值。
输出描述 Output Description
输出包括一行,这一行只包含一个整数,表示在规定的时间内,可以采到的草药的最大总价值。
样例输入 Sample Input
70 3
71 100
69 1
1 2
样例输出 Sample Output
3
数据范围及提示 Data Size & Hint
【数据规模】
对于30%的数据,M<=10;
对于全部的数据,M<=100。
典型的01背包问题,由于数据氛围较小,所以暴力应该也可以过
#include
#include
using namespace std;
int t,m,v[1001],w[1001],f[1001];
main()
{
scanf("%d%d",&t,&m);
for (int i=1;i<=m;i++)
scanf("%d%d",&w[i],&v[i]);//w表示代价,v表示价值
for (int i=1;i<=m;i++)
for (int j=t;j>=1;j--)
if (j-w[i]>=0) f[j]=max(f[j],f[j-w[i]]+v[i]);
else f[j]=f[j];//这一句并没有什么卵用
printf("%d",f[t]);
}
————————————————————————
SECOND
————————————————————————
学生在我们USACO的竞赛中的得分越多我们越高兴。我们试着设计我们的竞赛以便人们能尽可能的多得分,这需要你的帮助。我们可以从几个种类中选取竞赛的题目,这里的一个”种类”是指一个竞赛题目的集合,解决集合中的题目需要相同多的时间并且能得到相同的分数。
你的任务是写一个程序来告诉USACO的职员,应该从每一个种类中选取多少题目,使得解决题目的总耗时在竞赛规定的时间里并且总分最大。
你的程序应该确定我们应该从每个”种类”中选多少道题目使得能在竞赛的时间中得到最大的分数。来自任意的”种类”的题目数目可能任何非负数(0或更多)。计算可能得到的最大分数。
INPUT FORMAT:第 1 行:M, N–竞赛的时间和题目”种类”的数目。
第 2-N+1 行: 两个整数:每个”种类”题目的分数和耗时。
SAMPLE INPUT (file inflate.in) 300 4
100 60
250 120
120 100
35 20
OUTPUT FORMAT:单独的一行包括那个在给定的限制里可能得到的最大的分数。
605
{从第2个”种类”中选两题第4个”种类”中选三题}
完全背包,数据范围比较小
#include
#include
using namespace std;
int m,n,w[10001],v[10001],f[10001];
main()
{
scanf("%d%d",&m,&n);
for (int i=1;i<=n;i++) scanf("%d%d",&v[i],&w[i]);
for (int i=1;i<=n;i++)
for (int j=0;j<=m;j++)
if (j-w[i]>=0) f[j]=max(f[j],f[j-w[i]]+v[i]);
else f[j]=f[j];
printf("%d",f[m]);
}
————————————————————————
LAST
————————————————————————
描述 Description
暗黑游戏中,装备直接决定玩家人物的能力。可以使用Pg和Rune购买需要的物品。暗黑市场中的装备,每件有不同的价格(Pg和Rune)、能力值、最大可购买件数。Kid作为暗黑战网的一个玩家,当然希望使用尽可能少的Pg和Rune购买更优的装备,以获得最高的能力值。请你帮忙计算出现有支付能力下的最大可以获得的能力值。
输入格式 Input Format
第一行,三个整数N,P,R,分别代表市场中物品种类,Pg的支付能力和Rune的支付能力。
第2..N+1行,每行四个整数,前两个整数分别为购买此物品需要花费的Pg,Rune,第三个整数若为0,则说明此物品可以购买无数件,若为其他数字,则为此物品可购买的最多件数(S),第四个整数为该装备的能力值。
输出格式 Output Format
仅一行,一个整数,最大可获得的能力值。
样例输入 Sample Input
3 10 10 5 3 0 110 4 3 4 120 2 3 1 130
样例输出 Sample Output
370
时间限制 Time Limitation
各个测试点1s
对于100%的数据, N在1-150,P在1-100 , R<=100且为正整数, 0<=S<=32;
【样例解释】
选第二种装备2件和第三种装备1件。
混合背包,但是有双代价的存在,所以我们开一个二维数组分别存储比较好。
#include
#include
using namespace std;
int n,p,r,w1[20000],w2[20000],s[20000],v[20000],f[5000][5000];//w1.w2指代双价值pg与rune
main()
{
freopen("pgrune.in","r",stdin);
freopen("pgrune.out","w",stdout);
scanf("%d%d%d",&n,&p,&r);
int j,k;
for (int i=1;i<=n;i++) scanf("%d%d%d%d",&w1[i],&w2[i],&s[i],&v[i]);
for (int i=1;i<=n;i++)
if (s[i]==0)//完全背包
for (j=w1[i];j<=p;j++)
for (k=w2[i];k<=r;k++)
f[j][k]=max(f[j][k],f[j-w1[i]][k-w2[i]]+v[i]);
else if (s[i]==1)//01背包
for (j=p;j>=w1[i];j--)
for (k=r;k>=w2[i];k--)
f[j][k]=max(f[j][k],f[j-w1[i]][k-w2[i]]+v[i]);
else //混合背包
for (j=p;j>=0;j--)
for (k=r;k>=0;k--)
for (int l=0;l<=s[i];l++)
if (j-l*w1[i]<0||k-l*w2[i]<0) break;
else f[j][k]=max(f[j][k],f[j-l*w1[i]][k-l*w2[i]]+l*v[i]);
printf("%d",f[p][r]);
}