• 多重背包问题


    题目描述

    N N N 种物品和一个容量是 V V V 的背包。

    i i i 种物品最多有 s i s_i si 件,每件体积是 v i v_i vi,价值是 w i w_i wi

    求解将哪些物品装入背包,可使物品体积总和不超过背包容量,且价值总和最大。
    输出最大价值。

    输入格式

    第一行两个整数, N N N V V V,用空格隔开,分别表示物品种数和背包容积。

    接下来有 N N N 行,每行三个整数 v i v_i vi w i w_i wi s i s_i si,用空格隔开,分别表示第 i i i 种物品的体积、价值和数量。

    输出格式

    输出一个整数,表示最大价值。

    数据范围

    0 < N , V ⩽ 100 0 < N,V\leqslant100 0<N,V100
    0 < v i , w i , s i ⩽ 100 00<vi,wi,si100

    输入样例
    4 5
    1 2 3
    2 4 1
    3 4 3
    4 5 2
    
    • 1
    • 2
    • 3
    • 4
    • 5
    输出样例:
    10
    
    • 1

    动态规划回顾

    在解决这道题之前我们先来回顾一下 Acwing 的讲师:y总,他讲述的解决动态规划问题的一般步骤 。

    img

    • 集合:表示状态中每一个下标位置可能的选择。一维数组也好二维数组也罢,动态规划处理之后里面存储的元素就是这个状态下对应的最终结果。而这个结果的产生,就是集合中满足题意的那个元素。

    • 属性:属性需要根据题意来选择。就拿本题来说,要计算价值的最大值,那么属性就是集合中价值的最大值!

    • 状态计算:将每一个状态中的集合进行划分,根据集合的划分推出状态转移方程。

    • 集合划分的依据:划分出来的所有集合的并集不得遗漏一个状态中的任何选择。但是可以重复。

    朴素解法

    读完题目,灵光一闪,思路乍现。这不就是完全背包问题的一个特殊情况嘛,直接照抄完全背包问题朴素解法的思路,加上一个物品数量的限制就行啦!

    img

    根据完全背包问题的图解,多重背包问题中的 K 替换成每个物品的最大数量就 OK 了。

    下面我们来推导状态转移方程:

    我们不妨假设第 i 个物品我们选择了 k 个。

    • 当 k = 0 时,说明不选择第 i 个物品,此时 f [i, j] = f [i - 1, j],这倒是很简单!

    • 当 k 不等于 0 时该怎么办呢?同样借鉴一下 01 背包问题的想法:之前假设第 i 个物品选择了 k 个,我们可以先不看第 i 个物品,那么问题就变成了从 1 到 (i - 1) 中的物品中做选择,但是选择的体积还是不大于 j 吗?当然不是,因为我们忽略了第 i 个物品,并且我们选择了 k 个第 i 个物品,因此选择的总体积应该是不大于 j - k * v[i],(v[i] 是第 i 个物品的体积)。

    那么当 k 不等于 0 时,状态转移方程就可以写成: f [ i , j ] = f [ i − 1 ] [ j − k ∗ v [ i ] ] + k ∗ w [ i ] f[i, j] = f[ i - 1][ j - k * v[i] ] + k * w[i] f[i,j]=f[i1][jkv[i]]+kw[i]。这里为什么要加上 k ∗ w [ i ] k * w[i] kw[i] 呢?因为 f [ i − 1 ] [ j − k ∗ v [ i ] ] f [ i - 1][ j - k * v[i] ] f[i1][jkv[i]] 是忽略了第 i i i 个物品的选择的价值最大值,想要计算 f [ i , j ] f [ i, j ] f[i,j] 肯定要算上第 i 个物品的选择嘛!

    但是 k k k 肯定不能乱取:

    • 首先 k k k 的取值不能大于给定的物品的数量。
    • 其次 k ∗ v [ i ] k* v[i] kv[i] 必须要小于背包的总体积。

    因为要求的是价值的最大值,因此在遍历 k 的取值时要取价值最大的那一个!
    k = 0 k = 0 k=0 k ≠ 0 k \ne 0 k=0 的情况和完全背包问题一样是可以合并的。

    于是我们轻松写出了解决这道题的代码:

    #include 
    #include 
    
    using namespace std;
    
    const int N = 110;
    
    int n, m;
    int v[N], w[N], s[N];
    int f[N][N];
    
    int f[N][N];
    
    int main()
    {
        cin >> n >> m;
        for(int i = 1; i <= n; i++) cin >> v[i] >> w[i] >> s[i];
    
        for(int i = 1; i <= n; i++)
            for(int j = 0; j <= m; j++)
                for(int k = 0; k <= s[i] && k * v[i] <= j; k++)
                    f[i][j] = max(f[i][j], f[i - 1][j - v[i] * k] + w[i] * k);
    
        cout << f[n][m] << endl;
    
        return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27

    二进制优化

    5. 多重背包问题 II - AcWing题库

    这个题目数据范围更大,朴素解法会超时的。我们需要考虑优化朴素解法。

    我们先来看看能否使用完全背包问题的思路来优化多重背包问题:

    多重背包问题的状态转移方程展开:

    f [ i , j ] = m a x ( f [ i − 1 , j ] , f [ i − 1 , j − v i ] + w i , f [ i − 1 ] [ j − 2 ∗ v i ] + 2 ∗ w i , . . . , f [ i − 1 , j − s i ∗ v i ] + s i ∗ w i ) f[i,j] = max(f[i-1,j],f[i-1,j-v_i]+w_i,f[i-1][j-2*v_i]+2*w_i,...,f[i-1,j-s_i*v_i]+s_i*w_i) f[i,j]=max(f[i1,j],f[i1,jvi]+wi,f[i1][j2vi]+2wi,...,f[i1,jsivi]+siwi)

    f [ i , j − v i ] = m a x ( f [ i − 1 , j − v i ] , f [ i − 1 , j − 2 ∗ v i ] + w i , . . . , f [ i − 1 , j − s i ∗ v i ] + ( s i − 1 ) ∗ w i ) , f [ i − 1 , j − ( s i + 1 ) ∗ v i ] + s i ∗ w i f[i,j-v_i]=max(f[i-1,j-v_i],f[i-1,j-2*v_i]+w_i,...,f[i-1,j-s_i*v_i]+(s_i-1)*w_i),f[i-1,j-(s_i+1)*v_i] + s_i *w_i f[i,jvi]=max(f[i1,jvi],f[i1,j2vi]+wi,...,f[i1,jsivi]+(si1)wi),f[i1,j(si+1)vi]+siwi

    明显看到 f [ i , j − v i ] f[i,j-vi] f[i,jvi] 最后面多了一项 f [ i − 1 , j − ( s i + 1 ) ∗ v i ] + s i ∗ w i f[i-1,j-(s_i+1)*v_i] + s_i *w_i f[i1,j(si+1)vi]+siwi 显然是不能利用完全背包问题的优化思路来优化多重背包问题的。

    二进制优化

    我们先从具体的例子来看:假设第 i i i 个物品最多可以选择 1023 1023 1023 个,朴素解法中我们是从 0 − 1023 0-1023 01023 枚举第 i i i 个物品选择的个数。实际上,并不需要这么暴力枚举。

    我们可以对 0 − 1023 0-1023 01023 的枚举进行拆分: 1 , 2 , 4 , 8 , 16 , 32 , 64 , 128 , 256 , 512 1,2,4,8,16,32,64,128,256,512 1,2,4,8,16,32,64,128,256,512,拆分成 2 0 , 2 1 , 2 2 . . . 2^0,2^1,2^2 ... 20,21,22... 然后用拆分之后的数组合出来 0 − 1023 0-1023 01023 的枚举:

    • 只选择 1,我们可以做到枚举 0 − 1 0-1 01 个第 i i i 个物品。

    • 在选择 1 的基础上加上 2,就做到可以枚举 2 - 3 个第 i i i 个物品。加上仅选择 1 的情况,就能做到枚举 0 − 3 0-3 03 个第 i i i 个物品。

    • 在上一次的基础上加上 4,就能做到枚举 4 − 7 4-7 47 个第 i i i 个物品。加上上一次的结果 0 − 3 0-3 03,就能做到枚举 0 − 7 0-7 07 个第 i i i 个物品。

    • 在上一次的基础上加上 8,就能做到枚举 8 − 15 8-15 815 个第 i i i 个物品。加上上一次的结果 0 − 7 0-7 07,就能做到枚举 0 − 15 0-15 015 个第 i i i 个物品。

    • 以此类推,通过 1 , 2 , 4 , 8 , 16 , 32 , 64 , 128 , 256 , 512 1,2,4,8,16,32,64,128,256,512 1,2,4,8,16,32,64,128,256,512 的组合就能做到枚举 0 − 1023 0-1023 01023 个物品。

    你可能会说,1023 恰好是 2 10 − 1 2^{10-1} 2101 比较特殊,要不是这么特殊的数字也能进行这么拆分嘛?再来看一个例子:假设第 i i i 个物品可以选择 200 200 200 个。对 200 200 200 进行拆分: 1 , 2 , 4 , 8 , 16 , 32 , 64 , 128 1,2,4,8,16,32,64,128 1,2,4,8,16,32,64,128 能拆分到 128 128 128 吗?显然是不能的,如果拆分到 128 128 128 那么按照上面的策略进行组合,枚举的范围就是: 0 − 255 0-255 0255 显然超出了 0 − 200 0-200 0200 的范围。因此最后一个数不是 128 128 128 而是 200 − ( 1 + 2 + 4 + 8 + 16 + 32 + 64 ) = 73 200-(1+2+4+8+16+32+64) = 73 200(1+2+4+8+16+32+64)=73。我们来验证对不对哈:

    • 只选择 1,我们可以做到枚举 0 − 1 0-1 01 个第 i i i 个物品。

    • 在选择 1 的基础上加上 2,就做到可以枚举 2 - 3 个第 i i i 个物品。加上仅选择 1 的情况,就能做到枚举 0 − 3 0-3 03 个第 i i i 个物品。

    • 在上一次的基础上加上 4,就能做到枚举 4 − 7 4-7 47 个第 i i i 个物品。加上上一次的结果 0 − 3 0-3 03,就能做到枚举 0 − 7 0-7 07 个第 i i i 个物品。

    • 在上一次的基础上加上 8,就能做到枚举 8 − 15 8-15 815 个第 i i i 个物品。加上上一次的结果 0 − 7 0-7 07,就能做到枚举 0 − 15 0-15 015 个第 i i i 个物品。

    • 在上一次的基础上加上 16,就能做到枚举 16 − 31 16-31 1631 个第 i i i 个物品。加上上一次的结果 0 − 15 0-15 015,就能做到枚举 0 − 31 0-31 031 个第 i i i 个物品。

    • 在上一次的基础上加上 32,就能做到枚举 32 − 63 32-63 3263 个第 i i i 个物品。加上上一次的结果 0 − 31 0-31 031,就能做到枚举 0 − 63 0-63 063 个第 i i i 个物品。

    • 在上一次的基础上加上 64,就能做到枚举 64 − 127 64-127 64127 个第 i i i 个物品。加上上一次的结果 0 − 63 0-63 063,就能做到枚举 0 − 127 0-127 0127 个第 i i i 个物品。

    • 在上一次的基础上加上 73,就能做到枚举 73 − 200 73-200 73200 个第 i i i 个物品。加上上一次的结果 0 − 127 0-127 0127,就能做到枚举 0 − 200 0-200 0200 个第 i i i 个物品。

    答案完全没问题哈!

    接下来我们就可以总结一般规律

    假设第 i i i 个物品最大选择 s s s 个,那么我们可以拆分出以下数字: 2 0 , 2 1 , 2 2 , ⋅ ⋅ ⋅ , 2 k , c 2^0,2^1,2^2,···,2^k,c 20,21,22,⋅⋅⋅,2k,c 使得 2 0 + 2 1 + 2 2 + ⋅ ⋅ ⋅ + 2 k + c = s 2^0+2^1+2^2+···+2^k+c = s 20+21+22+⋅⋅⋅+2k+c=s。在拆分出来的这些数中选择特定的一些数,能够拼凑出 0 − s 0-s 0s 中的任意一个数。可以证明 0 ≤ c < 2 k + 1 0\le c < 2^{k+1} 0c<2k+1

    • c c c 等于 0 0 0 说明 s s s 比较特殊,恰好是 2 k + 1 − 1 2^{k+1}-1 2k+11,见第一个例子中 1023 的例子, c c c 还可以取其他小于 2 k + 1 2^{k+1} 2k+1 的整数。
    • 如果 c = 2 k + 1 c=2^{k+1} c=2k+1 ,那么我们可以将 c c c 拆出来一个 2 k + 1 2^{k+1} 2k+1,然后 c c c 变成 0。

    2 0 + 2 1 + 2 2 + ⋅ ⋅ ⋅ + 2 k = 2 k + 1 − 1 2^0+2^1+2^2+···+2^k = 2^{k+1} - 1 20+21+22+⋅⋅⋅+2k=2k+11 可以拼凑出 0 − ( 2 k + 1 − 1 ) 0-(2^{k+1}-1) 0(2k+11),加上 c c c 能够拼凑出: c − ( ( 2 k + 1 − 1 ) + c ) ) c - ((2^{k+1}-1) + c)) c((2k+11)+c)) 也就是 c − s c-s cs,现证明 0 − ( 2 k + 1 − 1 ) 0-(2^{k+1}-1) 0(2k+11) c − s c-s cs 能拼凑出 0 − s 0-s 0s

    • 想要证明 0 − ( 2 k + 1 − 1 ) 0-(2^{k+1}-1) 0(2k+11) c − s c-s cs 能拼凑出 0 − s 0-s 0s,只需要证明 ( 2 k + 1 − 1 ) + 1 ≥ c (2^{k+1}-1) + 1 \geq c (2k+11)+1c 即可,即证明: 2 k + 1 ≥ c 2^{k+1} \geq c 2k+1c
    • 我们在最开始已经证明了: 0 ≤ c < 2 k + 1 0\le c < 2^{k+1} 0c<2k+1,显然 2 k + 1 ≥ c 2^{k+1} \geq c 2k+1c 成立。

    因此如此拆分,能够拼凑出 0 − s 0-s 0s 中的任意一个数。

    这样拆分出来一些数之后有什么用呢?仔细想想:我们从拆分出来的数中选择若干个,能够凑出 0 − s 0-s 0s 中的任意一个数,这不就是 01 01 01 背包问题嘛!通过这些数字的组合,达到了枚举 0 − s 0-s 0s 的目的。

    朴素解法的时间复杂度: O ( N ∗ M ∗ S ) O(N*M*S) O(NMS)

    二进制优化之后的时间复杂度: O ( N ∗ M ∗ l o g 2 s ) O(N*M*log_2s) O(NMlog2s)

    下面就是代码实现啦!

    #include 
    #include 
    
    using namespace std;
    
    const static int N = 11010;
    
    int n, m;
    int v[N], w[N];
    int f[N]; //01背包问题可以直接优化到一维嘛
    
    
    int main()
    {
        cin >> n >> m;
    
        int cnt = 0;
        for(int i = 1; i <= n; i++)
        {
            int a, b, s;
            cin >> a >> b >> s;
    
            int k = 1;
            while(k <= s)
            {
                cnt++;
                v[cnt] = a * k; //拆出来的每个数 k 代表选择了 k 个第 i 个物品,对应的体积就是 a * k 
                w[cnt] = b * k; //拆出来的每个数 k 代表选择了 k 个第 i 个物品,对应的价值就是 b * k
                s -= k;
                k *= 2; 
            }
    
            if(s > 0) //剩下的 s 就是上面分析问题中的 c
            {
                cnt++;
                v[cnt] = a * s;
                w[cnt] = b * s; 
            }
        }
    
        n = cnt;
    
        //01 背包问题
        for(int i = 1; i <= n; i++)
        {
            for(int j = m; j >= v[i]; j--)
            {
                f[j] = max(f[j], f[j - v[i]] + w[i]);
            }
        }
    
        cout << f[m] << endl;
    
        return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55

    在这里插入图片描述

  • 相关阅读:
    【C++设计模式】(五)创建型模式 — 建造者模式
    数据可视化(七):Pandas香港酒店数据高级分析,涉及相关系数,协方差,数据离散化,透视表等精美可视化展示
    Jenkins+K8s实现持续集成(一)
    ESP32蓝牙实例-ESP32之间通过蓝牙串口主从机通信
    基于MATLAB的混沌数字图像加密技术研究与仿真实现
    C++学习笔记01-入门基础
    网站怎么防止ddos攻击,防御ddos攻击的11种方法
    Mybatis分页
    kaggle竞赛实战6——方案优化之交叉验证
    React 轮播图
  • 原文地址:https://blog.csdn.net/m0_73096566/article/details/134289917