背包问题poj1742

背包问题可以描述为:给定一组物品,每种物品都有自己的重量和价格,在限定的总重量内,我们如何选择,才能使得物品的总价格最高。可以分为01背包、完全背包和多重背包问题。

01 背包

有n 种不同的物品,每个物品有两个属性,size 体积,value 价值,现在给一个容量为 w 的背包,问最多可带走多少价值的物品。  

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. int f[w+1];   //f[x] 表示背包容量为x 时的最大价值  
  2. for (int i=0; i<n; i++)  
  3.     for (int j=w; j>=size[i]; j--)  
  4.         f[j] = max(f[j], f[j-size[i]]+value[i]);  
这是最基础的背包问题,特点是:每种物品仅有一件,可以选择放或不放。
用子问题定义状态:即f[i][v]表示前i件物品恰放入一个容量为v的背包可以获得的最大价值。则其 状态转移方程便是:f[i][v]=max{ f[i-1][v], f[i-1][v-w[i]]+v[i] }。可以压缩空间,f[v]=max{f[v],f[v-w[i]]+v[i]}
这个方程非常重要,基本上所有跟背包相关的问题的方程都是由它衍生出来的。 所以有必要将它详细解释一下:“将前i件物品放入容量为v的背包中”这个子问题,若只考虑第i件物品的策略(放或不放),那么就可以转化为一个只牵扯前i-1件物品的问题。如果不放第i件物品,那么问题就转化为“前i-1件物品放入容量为v的背包中”,价值为f[i-1][v];如果放第i件物品,那么问题就转化为“前i-1件物品放入剩下的容量为v-w[i]的背包中”,此时能获得的最大价值就是f [i-1][v-w[i]]再加上通过放入第i件物品获得的价值v[i]。
注意f[v]有意义当且仅当存在一个前i件物品的子集,其费用总和为v。所以按照这个方程递推完毕后,最终的答案并不一定是f[N] [V],而是f[N][0..V]的最大值。如果将状态的定义中的“恰”字去掉,在转移方程中就要再加入一项f[v-1],这样就可以保证f[N] [V]就是最后的答案。至于为什么这样就可以,由你自己来体会了。

完全背包 

如果物品不计件数,就是每个物品不只一件的话,稍微改下即可  

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. for (int i=0; i<n; i++)  
  2.     for (int j=size[i]; j<=w; j++)  
  3.         f[j] = max(f[j], f[j-size[i]]+value[i]);  
多重背包 :  
         多重背包问题要求很简单,就是每件物品给出确定的件数,求 可得到的最大价值  

下面是一个多重背包问题的例子poj1742


Description

People in Silverland use coins.They have coins of value A1,A2,A3...An Silverland dollar.One day Tony opened his money-box and found there were some coins.He decided to buy a very nice watch in a nearby shop. He wanted to pay the exact price(without change) and he known the price would not more than m.But he didn't know the exact price of the watch. 
You are to write a program which reads n,m,A1,A2,A3...An and C1,C2,C3...Cn corresponding to the number of Tony's coins of value A1,A2,A3...An then calculate how many prices(form 1 to m) Tony can pay use these coins. 

Input

The input contains several test cases. The first line of each test case contains two integers n(1<=n<=100),m(m<=100000).The second line contains 2n integers, denoting A1,A2,A3...An,C1,C2,C3...Cn (1<=Ai<=100000,1<=Ci<=1000). The last test case is followed by two zeros.

Output

For each test case output the answer on a single line.

Sample Input

3 10
1 2 4 2 1 1
2 5
1 4 2 1
0 0
<span style="font-family: Arial; background-color: rgb(240, 240, 240);">Sample Output</span>
8
    4
 
#include <iostream>
#include<cstdio>
#include<cstring>
#include<string>
#include<algorithm>
using namespace std;


int main()
{
    int n,m,i,j,sum;
    int a[100001],c[1001],ans[100001],f[1000001];
    while(cin>>n>>m&&(n!=0||m!=0))
          {
             for(i=1;i<=n;i++)
                {
                    cin>>a[i];
                }
             for(i=1;i<=n;i++)
                {
                    cin>>c[i];
                }
            memset(f,0,sizeof(f));
            f[0]=1;
            sum=0;
            for(i=1;i<=n;i++)
            {
                memset(ans,0,sizeof(ans));
                for(j=a[i];j<=m;j++)
                {
                    if(!f[j]&&f[j-a[i]]&&ans[j-a[i]]+1<=c[i])
                    {
                        f[j]=1;
                        ans[j]=ans[j-a[i]]+1;
                        sum++;
                    }
                }
            }
            cout<<sum<<endl;
          }
    return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值