算法分析与设计——过载问题(0-1背包)


一、问题

在这里插入图片描述

二、解析

有n件物品,每件物品的重量为w[i],价值为c[i]。现有一个容量为V的背包,问如何选取物品放入背包,使得背包内物品的总价值最大。其中
每种物品都只有一件。

令dp[i][j]来表示前i件物品装入容量为j的背包所能得到的最大总价值。

对于dp[i][j]来说,i指的是前i件物品,j指的是还剩下多少背包空间。

对于一件物品,我们有放进或者不放进背包两种选择:

  1. 假如我们放进背包,f[i][j] = f[i - 1][j - weight[i]] + value[i],这里的f[i - 1][j - weight[i]] + value[i]应该这么理解:在没放这件物品之前的状态值加上要放进去这件物品的价值。而对于f[i - 1][j - weight[i]]这部分,i - 1很容易理解,关键是 j - weight[i]这里,我们要明白:要把这件物品放进背包,就得在背包里面预留这一部分空间;
  2. 假如我们不放进背包,f[i][j] = f[i - 1][j],这个很容易理解;
  3. 背包放不下当前这一件物品,这种情况下f[i][j] = f[i - 1][j]。

因此,我们的状态转移方程就是:f[i][j] = max(f[i][j] = f[i - 1][j] , f[i - 1][j - weight[i]] + value[i])

三、设计

    for (int i = 1; i <= n; i++) 
    {
        for (int j = 1; j <= b; j++) 
        {
            if (w[i] > j) f[i][j] = f[i - 1][j];
            else f[i][j] = max(f[i - 1][j], f[i-1][j - w[i]] + v[i]);
        }
    }

四、分析

时间复杂度为O(nlogn)

五、代码

#include<iostream>
#include<vector>
#include<cmath>
#include<time.h>
#include<cstring>
#include<algorithm>
 
using namespace std;

#define N 20
int f[N+2][N+2];//全局变量,自动初始化为0  
int w[N];  
int v[N];  
int main(){  
    int n,b;
    cout << "请输入物品个数" << endl;
    cin>>n;//物品种类
    cout << "请输入背包最大容量" << endl;
    cin>>b;//背包容量
    cout << "请输入每个物品的重量和价值" << endl;
    for (int i=1;i<=n; i++)  
    {  
        cin >> w[i] >> v[i];  
    }  
    for (int i = 1; i <= n; i++) 
    {
		for (int j = 1; j <= b; j++) 
        {
			if (w[i] > j) f[i][j] = f[i - 1][j];
			else f[i][j] = max(f[i - 1][j], f[i-1][j - w[i]] + v[i]);
		}
	}
    cout << "当装"<<n<< "种物品时的最大价值是" <<f[n][b] <<endl;//输出最优解
    system("pause");
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值