背包2
Time Limit: 2000/1000ms (Java/Others)
Problem Description:
有n个重量和价值分别为Wi,Vi的物品,现从这些物品中挑选出总量刚好为 W 的物品,求所有方案中价值总和的最大值。
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 }