多重背包问题

题目描述

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
输出样例:
10

动态规划回顾

在解决这道题之前我们先来回顾一下 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;
}

二进制优化

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张图片

你可能感兴趣的:(数据结构与算法,c++,开发语言,算法,数据结构)