• ACM_01背包(恰好装满)


    背包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 }
    
    
  • 相关阅读:
    剑指Offer34 数组中的逆序对
    剑指Offer33 第一个只出现一次的字符
    剑指Offer32 丑数
    剑指Offer31 把数组排成最小的数
    剑指Offer30 从1到n整数出现1的个数
    剑指Offer29 连续子数组最大和
    剑指Offer28 最小的K个数(Partition函数应用+大顶堆)
    DNS的递归查询和迭代查询
    剑指Offer27 数组中超过一半的数
    剑指Offer26 字符串的全排列
  • 原文地址:https://www.cnblogs.com/acgoto/p/8999316.html
Copyright © 2020-2023  润新知