1.    0,1问题

一个背包,往里装东西,重量w(weight)分别为为[2,3,4,5] 价值v(value)对应为[3,4,5,6] 如果你的容量为8,每个物品只有一个,求你能装入背包的最大价值

我们可以一步一步来 ,先创建一个表格 (数组), 数组dp[i][j] i代表你只用前i个物体,j代表你的剩余容量,dp[i][j] 代表在此情况下可以得到的最大价值,先得到一个表格

背包问题 数量有上限 python_数据结构与算法

先填第一行(也就是假设现在你只有第一个物体), 首先第一个物体重量为2 剩余容量为1,剩余容量小于物体重量,则最大价值为0, 第二个数剩余容量=物体容量,可以拿 填上第一个物体的价值3 就这样填上第一行 , 继续填第二行 第二行代表的意义就是我们现在可以使用的有两个物体了,使用这两个物体能够得到的最大价值。这一共可以分为两种情况,一种是放第二个物体,一种是不放第二个物体,选择其中大的,转移方程为dp[i][j] = max(dp[ i - 1, j ], dp[ i - 1, j - w[ i ] ] + v [ i ] ) 这里解释下放第二个物体的式子,dp[ i-1, j - w[ i ] ]代表你把第二个物体的容量空出来时,能得到的最大价值, 再加上这个物体的价值。就这样填好所有的。

背包问题 数量有上限 python_数据结构与算法_02

则 表格中最后一个物体就是面对所有物体,剩余容量为C时能得到的最大价值了。

def pack1(w, v, C): #每个东西只能选择一次
    dp = [[0 for _ in range(C+1)] for _ in range(len(w)+1)]
    for i in range(1, len(w)+1):
        for j in range(1, C+1):
            if j < w[i-1]: #如果剩余容量不够新来的物体 直接等于之前的
                dp[i][j] = dp[i-1][j]
            else:
                dp[i][j] = max(dp[i-1][j], dp[i-1][j-w[i-1]]+ v[i-1])
    return dp[len(w)][c]

背包问题 数量有上限 python_sed_03

这个是得到的dp结果

观察发现,可以对它的空间复杂度进行优化,我们可以用一维数组只保存最后一行, 代码如下

def pack1(w, v, c):
    #它是先得到第一行的值,存到dp中,然后再直接用dp相当于就是上一行的值,所以下面必须用逆序
    #否则dp[j-w[i-1]]可能会用到你本行的值,从大到小就不会
    dp = [0 for _ in range(c+1)]
    for i in range(1, len(w)+1):
        for j in reversed(range(1, c+1)):#这里必须用逆序
            if w[i-1] <= j:
                dp[j] = max(dp[j], dp[j-w[i-1]]+v[i-1])
    return dp[c]

2 背包问题变形:完全背包问题

还是原来的问题,上次是每个物体只能用1次,现在改成无限次,转移方程只需要修改一点即可dp[i][j] = max(dp[ i - 1, j ], dp[ i, j - w[ i ] ] + v [ i ] )  只需要把后面的i-1 -> i 即可

def pack2(w, v, C): #每个东西能选择多次 完全背包问题
    dp = [[0 for _ in range(C+1)] for _ in range(len(w)+1)]
    for i in range(1, len(w)+1):
        for j in range(1, C+1):
            if j < w[i-1]:
                dp[i][j] = dp[i-1][j]
            else:
                dp[i][j] = max(dp[i-1][j], dp[i][j-w[i-1]]+ v[i-1])
    for i in dp:
        print(i)
pack2([2,3,4,5], [3,4,5,6], 8)

结果为

[0, 0, 0, 0, 0, 0, 0, 0, 0]
[0, 0, 3, 3, 6, 6, 9, 9, 12]
[0, 0, 3, 4, 6, 7, 9, 10, 12]
[0, 0, 3, 4, 6, 7, 9, 10, 12]
[0, 0, 3, 4, 6, 7, 9, 10, 12]

同样可以优化空间复杂度 ,只需要把逆序改成顺序 即可。

def pack2(w, v, C):    
    dp = [0 for _ in range(c+1)]
    for i in range(1, len(w)+1):
        for j in (range(1, c+1)):
            if w[i-1] <= j:
                dp[j] = max(dp[j], dp[j-w[i-1]]+v[i-1])
    return dp[c]

3 变形:多重背包问题,给定数量的物体,第一个是1个物体,第二个是无限物体,这个是介于之间,给定物体数目

方法1:再多加个循环试一下k个物体的价值, 这次直接上优化完的一维数组的

def pack3(w, v, s, c):
    dp = [0 for _ in range(c+1)]
    for i in range(1, len(w)+1):
        for j in reversed(range(1, c+1)):
            for k in range(s[i-1] + 1):
                if k*w[i-1] <= j:
                    dp[j] = max(dp[j], dp[j-k*w[i-1]]+k*v[i-1])
    return dp[c]

方法2:想象加入物体数量为4个我们直接在 w= [2,3,4,5]  v = [3,4,5,6]  s(个数)=[1,1,1,3] 如果我们直接把4个物体分成3个1个物体,就可以直接转变成0,1问题 。 w变成[2,3,4,5,5,5], v变成[3,4,5,6,6,6] 这就完全变成了0,1问题。  但是如果物体个数太多这样复杂度 高,我们有更好的划分方法。 比如13 我们可以划分为1+2+4+6,利用这个划分,原来需要划分为n个1,现在只需要log n 个组 了

def pack3(w, v, s, c):
	for i in range(len(s)):
	    k = 1
	    s_value = s[i]
	    while k<=s_value:
	        w2.append(k*w[i])
	        v2.append(k*v[i])
	        s_value -= k
	        k *= 2
	    if s_value>0:
	        w2.append(s_value*w[i])
	        v2.append(s_value*v[i])  
    #前面是划分,后面是0,1背包
	dp = [0 for _ in range(c+1)]
	for i in range(1, len(w2)+1):
	    for j in reversed(range(1, c+1)):
	        if w2[i-1] <= j:
	            dp[j] = max(dp[j], dp[j-w2[i-1]]+v2[i-1])
	return dp[c]

4 背包问题变形:混合背包问题

物品一共有三类:

  • 第一类物品只能用1次(01背包);
  • 第二类物品可以用无限次(完全背包);
  • 第三类物品最多只能用 sisi 次(多重背包);
  • si=−1 表示第 ii 种物品只能用1次;
  • si=0 表示第 ii 种物品可以用无限次;
  • si>0 表示第 ii 种物品可以使用 si 次;
def pack4(w, v, c, s):
	w2 = []
	v2 = []
	s2 = []
	for i in range(len(s)):
	    if s[i] == 0 or s[i] == -1:
	        w2.append(w[i])
	        v2.append(v[i])
	        s2.append(s[i])
	    else:
	        s_value = s[i]
	        k = 1
	        while k <= s_value:
	            w2.append(k*w[i])
	            v2.append(k*v[i])
	            s2.append(s[i])
	            s_value -= k
	            k *= 2
	        if s_value> 0:
	            w2.append(s_value*w[i])
	            v2.append(s_value*v[i])
	            s2.append(s[i])
	 #上面把si>0的背包拆分了(变成0,1背包)下面分成0,1背包和无限背包两种
	dp = [0 for _ in range(c+1)]
	for i in range(1, len(w2)+1):
	    if s2[i-1] == 0:
	        for j in (range(1, c+1)):
	            if j-w2[i-1]>=0:
	                dp[j] = max(dp[j], dp[j-w2[i-1]]+v2[i-1])
	    else:
	        # print('i',i)
	        for j in reversed(range(1, c+1)):
	            if j-w2[i-1]>=0:
	                # print('k', k)
	                dp[j] = max(dp[j], dp[j-w2[i-1]]+v2[i-1])
	                # print('dp['+str(j)+']', dp[j])
	    # print(dp)
	print(dp[c])

5 背包问题变形:分组背包问题

分组,组内只能选择一个

def pack5(w, v, n, c):
	"""
	w: list[list, list,...]
	v: list[list, list,...]
	n:list 组内的长度
	"""
	dp = [0 for _ in range(c+1)]
	for i in range(1, len(w)+1):
	    for j in reversed(range(1, c+1)):
	        for k in range(n_list[i-1]): #别的和0,1背包一样 就是这里枚举一下每个组内的值,在每个组内选出一个max值
	            if j-w[i-1][k] >= 0:
	                # print(dp[j])
	                dp[j] = max(dp[j], dp[j-w[i-1][k]] + v[i-1][k])
	    # print(dp)
	print(dp[c])

6 背包问题变形:二维费用背包问题

在重量的基础上再加上一个体积约束

新增leetcode相关题目

背包问题 数量有上限 python_背包问题 数量有上限 python_04