01背包问题(个人学习笔记)

新手第一次写博客,作为自己的学习笔记,如有错误之处还请各位大佬多多指出;


01背包问题介绍

「01背包问题」,是让你从N个物品中,选出总价值之和最大并且不超过背包总体积V的物品;(每一件物品都有自己的价值和相应的体积,并且没一件物品只可以选择一次)。

这是一个比较简单的动态规划的问题,动态规划是不断决策求最优解的过程,「0-1 背包」即是不断对第 i 个物品的做出决策,「0-1」正好代表不选与选两种决定。

(我这里使用的是二维的,之后会更新一维版本)


源代码:

#include <iostream>
#include <cstring>
#include <algorithm>

using namespace std;
const int N = 2010;
int w[N], v[N];  //v[i] 表示第i个物品的体积  w[i]表示第i个物品的价值
int dp[N][N];    //状态表示:dp[i][j] 前i个物品,在容积为j的情况下的最大价值

int main()
{
    int n, m;    //n:物品数量  m:背包容积
    
    cin >> n >> m;
    
    for(int i = 1; i <= n; i++)
        cin >> v[i] >> w[i];   //输入每一个物品的价值和相应的体积
    
    for(int i  = 1; i <= n; i ++)  
        for(int j = 1; j <= m; j++)  //「1」
        {
            //当前背包容量够装下第i个物品
            if(j >= v[i])
            {
                //「2」
                dp[i][j] = max(dp[i-1][j - v[i]] + w[i], dp[i - 1][j]); 
            }
            //当前背包容量不够装下第i个物品
            else
            {
                //「3」
                dp[i][j] = dp[i - 1][j];
            }
        }
    //直接输出,n个物品,体积为m时,最优情况;
    cout << dp[n][m];
    
    return 0;
}

推导过程:

「01背包问题」最重要的就是选择的过程了,这个过程我也是自己手推了一遍,才算真的明白了,接下来我写一下对于上面三个标注地方的理解:

「1」:每一次 j 都要从 1 开始,直到 m,这是因为 j 取的每一个数(体积)我们都要对其选择价值价值最大的情况,至于怎么判断,请看下面;

「2」:j >= v[ i ] : 当前体积 j 可以容纳下体积为 v[ i ] 的物品。

此时,我们所需要做得就是比较 dp[ i - 1 ][j - v[ i ] ] + w[ i ]   与  dp[ i - 1][ j ] 大小,并且取较大值;

dp[ i - 1 ][j - v[ i ] ] + w[ i ]  :表示前一个物品,占用的体积为 j - v[ i ] 时,所取得的值,加上当前选择的第 i 个物品的价值;(注意:上一个物品的最优值值随着 j 的改变,也是在发生变化的)

dp[ i - 1][ j ]:表示不取当前物品,并且前一个物品所占用容积为 时,所取得的值;

注意:「2」是我这个小菜鸡不容易理解的地方,所以我动手推了一下这个过程,结果恍然大悟,如果大家不明白的话,可以找一道例题,取动手写一下,相信你也会明白的;

「3」j < v [ i ] : 当前体积无法容纳第 i 个物品,不可以选择,所以此时价值最大就是前 i - 1个物品的最大值;


总结

这是我的一个学习笔记,希望各位大佬看了之后,能够给出一些建议,我这个小菜鸡一定会好好学的,谢谢!

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值