概述
背包2
Time Limit: 2000/1000ms (Java/Others)
Problem Description:
有n个重量和价值分别为Wi,Vi的物品,现从这些物品中挑选出总量刚好为 W 的物品,求所有方案中价值总和的最大值。
Input:
输入包含多组测试用例,每一例的开头为两位整数 n、W(1<=n<=10000,1<=W<=1000),接下来有 n 行,每一行有两位整数 Wi、Vi(1<=Wi<=10000,1<=Vi<=100)。
Output:
输出为一行,即所有方案中价值总和的最大值。若不存在刚好填满的情况,输出“-1”。
Sample Input:
3 4 1 2 2 5 2 1 3 4 1 2 2 5 5 1
Sample Output:
6 -1
解题思路:求01背包恰好装满时得到的最大价值,应该这样初始化:①dp[0]=0,表示背包容量为0时得到的最大价值也为0;②dp[1~W]=-inf,表示背包容量为其他状态下都为非法状态(未装满),因为我们还要求解恰好装满的情况下得到的最大价值。那么在求解过程中,由于动规的基本思想是将待求解问题分解成若干个子问题,先求解子问题,然后从这些子问题的解得到原问题的解,所以如果子问题状态是非法的(-inf),则当前问题的状态依然非法,即不存在恰好装满的情况;相反,如果子问题的状态是合法(不是-inf)的(恰好装满),那么当前问题求解也可得到合法状态。这样最后判断一下dp[W]是否恰好装满即dp[W]>=0,否则为未装满状态。
AC代码:
1 #include<bits/stdc++.h> 2 using namespace std; 3 int n,W,w[10005],v[10005],dp[1005]; 4 int main(){ 5 while(cin>>n>>W){ 6 memset(dp,-0x3f,sizeof(dp));dp[0]=0; 7 for(int i=1;i<=n;++i) 8 cin>>w[i]>>v[i]; 9 for(int i=1;i<=n;++i) 10 for(int j=W;j>=w[i];--j) 11 dp[j]=max(dp[j],dp[j-w[i]]+v[i]); 12 if(dp[W]<0)cout<<-1<<endl; 13 else cout<<dp[W]<<endl; 14 } 15 return 0; 16 }
转载于:https://www.cnblogs.com/acgoto/p/8999316.html
最后
以上就是顺利香水为你收集整理的ACM_01背包(恰好装满)的全部内容,希望文章能够帮你解决ACM_01背包(恰好装满)所遇到的程序开发问题。
如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。
本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
发表评论 取消回复