nyoj289苹果

来源:互联网 发布:全球网络实时监控 编辑:程序博客网 时间:2024/05/01 23:48

苹果

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

ctest有n个苹果,要将它放入容量为v的背包。给出第i个苹果的大小和价钱,求出能放入背包的苹果的总价钱最大值。


输入
有多组测试数据,每组测试数据第一行为2个正整数,分别代表苹果的个数n和背包的容量v,n、v同时为0时结束测试,此时不输出。接下来的n行,每行2个正整数,用空格隔开,分别代表苹果的大小c和价钱w。所有输入数字的范围大于等于0,小于等于1000。
输出
对每组测试数据输出一个整数,代表能放入背包的苹果的总价值。
样例输入
3 31 12 13 10 0
样例输出
2

 0-1背包问题,hoho~~

先说一下0-1背包

P01: 01背包问题 
题目 
N件物品和一个容量为V的背包。第i件物品的费用是c[i],价值是w[i]。求解将哪些物品装入背包可使这些物品的费用总和不超过背包容量,且价值总和最大。 

基本思路 
这是最基础的背包问题,特点是:每种物品仅有一件,可以选择放或不放。 

用子问题定义状态:即f[i][v]表示前i件物品恰放入一个容量为v的背包可以获得的最大价值。则其状态转移方程便是:f[i][v]=max{f[i-1][v],f[i-1][v-c[i]]+w[i]} 

这个方程非常重要,基本上所有跟背包相关的问题的方程都是由它衍生出来的。所以有必要将它详细解释一下:将前i件物品放入容量为v的背包中这个子问题,若只考虑第i件物品的策略(放或不放),那么就可以转化为一个只牵扯前i-1件物品的问题。如果不放第i件物品,那么问题就转化为i-1件物品放入容量为v的背包中;如果放第i件物品,那么问题就转化为i-1件物品放入剩下的容量为v-c[i]的背包中,此时能获得的最大价值就是f [i-1][v-c[i]]再加上通过放入第i件物品获得的价值w[i] 

注意f[i][v]有意义当且仅当存在一个前i件物品的子集,其费用总和为v。所以按照这个方程递推完毕后,最终的答案并不一定是f[N] [V],而是f[N][0..V]的最大值。如果将状态的定义中的字去掉,在转移方程中就要再加入一项f[i][v-1],这样就可以保证f[N] [V]就是最后的答案。至于为什么这样就可以,由你自己来体会了。 

优化空间复杂度 
以上方法的时间和空间复杂度均为O(N*V),其中时间复杂度基本已经不能再优化了,但空间复杂度却可以优化到O(V) 

先考虑上面讲的基本思路如何实现,肯定是有一个主循环i=1..N,每次算出来二维数组f[i][0..V]的所有值。那么,如果只用一个数组f [0..V],能不能保证第i次循环结束后f[v]中表示的就是我们定义的状态f[i][v]呢?f[i][v]是由f[i-1][v]f[i-1] [v-c[i]]两个子问题递推而来,能否保证在推f[i][v]时(也即在第i次主循环中推f[v]时)能够得到f[i-1][v]f[i-1][v -c[i]]的值呢?事实上,这要求在每次主循环中我们以v=V..0的顺序推f[v],这样才能保证推f[v]f[v-c[i]]保存的是状态f[i -1][v-c[i]]的值。伪代码如下: 

for i=1..N 
for v=V..0 
f[v]=max{f[v],f[v-c[i]]+w[i]}; 

其中的f[v]=max{f[v],f[v-c[i]]}一句恰就相当于我们的转移方程f[i][v]=max{f[i-1][v],f[i- 1][v-c[i]]},因为现在的f[v-c[i]]就相当于原来的f[i-1][v-c[i]]。如果将v的循环顺序从上面的逆序改成顺序的话,那么则成了f[i][v]f[i][v-c[i]]推知,与本题意不符,但它却是另一个重要的背包问题P02最简捷的解决方案,故学习只用一维数组解01背包问题是十分必要的。 

总结 
01
背包问题是最基本的背包问题,它包含了背包问题中设计状态、方程的最基本思想,另外,别的类型的背包问题往往也可以转换成01背包问题求解。故一定要仔细体会上面基本思路的得出方法,状态转移方程的意义,以及最后怎样优化的空间复杂度。 

  摘自:dd大牛的《背包九讲》:点击打开链接

好啦~说这道题目,每次苹果也都是两个选择,放或者不放~

代码:

 #include <iostream>#include <cstdio>#include <cstring>using namespace std;int main(){    int n, v;    int num[1001];    int c, vi;    while(scanf("%d%d", &n, &v)!=EOF && (n+v))    {        memset(num, 0, sizeof(num));        for(int i=1; i<=n; i++)        {            scanf("%d%d", &c, &vi);            for(int vj=v; vj>=c; vj--)  //注意vj的顺序应该是倒序的,正序会产生累加。完全背包再说。。。            {                num[vj] = max(num[vj], num[vj-c]+vi); //每次都在选这个和不选这个中间找到一个价值最大的保存            }        }        printf("%d\n", num[v]);    }    return 0;}        


0 0
原创粉丝点击