基本0 - 1背包问题

基本0 - 1 背包

状态表示

dp[i][j]:前i个物品,背包容量j下的最优解(最大价值)

对于基本01背包,我们主要的问题就是不拿
如果不能拿,那么就将上一状态转移到当前状态
dp[i][j] = dp[i - 1][j]
如果可以拿,我们就要比较拿或者是不拿哪一种价值最高

  1. 不拿当前物品
    dp[i][j] = dp[i - 1][j]
  2. 拿当前物品
    dp[i][j] = dp[i - 1][j - w[i]] + v[i]

不拿当前物品,那么当前状态是上一状态的转移
拿当前物品,我们只要确定来拿这件物品,那么说明当前的背包容量一定是可以装下,那么我们就要来转移一个减去当前物品体积的状态,然后在加上当前物品的价值。

则当前状态的最优解为
dp[i][j] = max(dp[i - 1][j],dp[i - 1][j - w[i]] + v[i])

题目

在这里插入图片描述

输入

4 5
1 2
2 4
3 4
4 5

输出

8

朴素代码

#include <iostream>
#include <algorithm>

using namespace std;

const int N = 1010;

int n,c;
int w[N];//每个物品的体积
int v[N];//每个物品的价值
int dp[N][N];// 前 i 个物品,背包容量 j 下的最优解(最大价值)

int main()
{
    cin >> n >> c;
    
    for(int i = 1;i <= n ;i ++)
        cin >> w[i] >> v[i];
    
    for(int i = 1;i <= n ;i ++)
    {
        for(int j = 0;j <= c; j++)
        {
        	if(j < w[i])
        	{
        		//当前物品不能拿时
            	dp[i][j] = dp[i - 1][j];//先把上一状态进行转移
            }
            else
            {
                //当前物品可以拿时,比较拿和不拿哪一个价值最优
                dp[i][j]=max(dp[i - 1][j],dp[i - 1][j - w[i]] + v[i]);
            }
        }
    }
    
    cout << dp[n][c] <<  endl;
    
    return 0;
}

优化

对于朴素代码,空间复杂度较高,面对数据很大的题目来说就可能爆内存
由朴素代码可知,当前状态只需要上一状态的转移,所以我们可以利用滚动数组的方法来降一维空间

优化代码

#include <iostream>
#include <algorithm>

using namespace std;

const int N = 1010;

int n,c;
int w[N];//每个物品的体积
int v[N];//每个物品的价值
int dp[N];// 前 i 个物品,背包容量 j 下的最优解(最大价值)

int main()
{
    cin >> n >> c;
    
    for(int i = 1;i <= n ;i ++)
        cin >> w[i] >> v[i];
    
    for(int i = 1;i <= n ;i ++)
    {
        for(int j = c;j >= w[i]; j --)
        {//应该是从后往前遍历,才可利用上一状态的数据
            dp[j]=max(dp[j],dp[j - w[i]] + v[i]);
        }
    }
    
    cout << dp[c] <<  endl;
    
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值