Python实现动态规划01背包问题

代码:https://blog.csdn.net/qq_34178562/article/details/79959380

理论讲解:

https://www.bilibili.com/video/BV1K4411X766?from=search&seid=13724711054890720316

https://www.bilibili.com/video/BV1XD4y1S7nv?from=search&seid=13724711054890720316

 

先假设一个数据输入:六个东西,东西A重2kg价值2毛钱,东西B重2kg价值3毛钱,东西C重3kg价值1毛钱,东西D重1kg价值5毛钱,东西E重5kg价值4毛钱,东西F重2kg价值3毛钱。问容量10kg书包最多能装几毛钱的东西。

这个问题需要建立一个动态规划表,能理解这个表的建立过程就能知道代码怎么写。

价值(毛)0kg1kg2kg3kg4kg5kg6kg7kg8kg9kg10kg
不放东西00000000000
A w=2 v=200222222222
B w=2 v=300335555555
C w=3 v=100335556666
D w=1 v=505588101010111111
E w=5 v=405588101010121214
F w=2 v=305588111113131315

首先,如果我们不放东西,背包肯定没有价值,所以不放东西这一排为0。背包容量为0,放不进去东西,背包也没有价值,所以0kg这一列为0。

然后,我们对应公式来看:V(i,j)表示将前i个物品放进容量为j的背包。

1)我们看A这一排,(A,1kg)等于0,为什么等于0,1kg小于A的重量2kg,符合j<wi,所以(A,1kg)等于(A,0kg),等于0。

2)我们看(A,2kg)等于2,为什么等于2,2kg等于A的重量2kg,符合j>=wi,所以(A,2kg)等于max( (不放东西,2kg), (不放东西,2kg-2kg)+A的价值 ),即max(0, 0+2)=2。

3)我们看A这一排的后面,由于i不变,max的前一项永远是(不放东西,*)=0,max的后一项永远是(不放东西,*)+A的价值=2,所以A这一排后面全是2。

细品

4)我们看(B,1kg)等于0,为什么等于0,1kg小于B的重量2kg,符合j<wi,所以(B,1kg)等于(B,0kg),等于0。

5)我们看(B,2kg)等于3,为什么等于3,2kg等于B的重量2kg,符合j>=wi,所以(B,2kg)等于max( (A,2kg), (A,2kg-2kg)+B的价值 ),即max(2, 0+3)=3。

6)我们看(B,4kg)等于5,为什么等于5,4kg大于B的重量2kg,符合j>=wi,所以(B,4kg)等于max( (A,4kg), (A,4kg-2kg)+B的价值 ),即max(2, 2+3)=5。

对比着品

7)我们看(D,1kg)等于5,为什么等于5,1kg等于D的重量1kg,符合j>=wi,所以(D,1kg)等于max( (C,1kg), (C,1kg-1kg)+D的价值 ),即max(0, 0+5)=5。

8)我们看(F,10kg),10kg大于F的重量2kg,符合j>=wi,所以(F,10kg)等于max( (E,10kg), (E,10kg-2kg)+F的价值 ),即max(14, 12+3)=15。

最后,问题的答案既是10kg背包能装最多1块5毛钱的东西。

 

保存一份代码,牢记公式,分清楚代码里面的是啥就行。

def bag(n, c, w, v):
    """
    测试数据:
    n = 6  物品的数量,
    c = 10 书包能承受的重量,
    w = [2, 2, 3, 1, 5, 2] 每个物品的重量,
    v = [2, 3, 1, 5, 4, 3] 每个物品的价值
    """
    #置零,表示初始状态
    value = [[0 for j in range(c + 1)] for i in range(n + 1)]
    for i in range(1, n + 1):
        for j in range(1, c + 1):

            #
            value[i][j] = value[i - 1][j]

            #
            if j >= w[i - 1] and value[i][j] < value[i - 1][j - w[i - 1]] + v[i - 1]:
                value[i][j] = value[i - 1][j - w[i - 1]] + v[i - 1]

    #打印动态规划表
    for x in value:
        print(x)

    return value

#w = list(map(int,input().split(' ')))
#v = list(map(int,input().split(' ')))
#c = eval(input())

w = [2, 2, 3, 1, 5, 2]
v = [2, 3, 1, 5, 4, 3]
c = 10
n = len(v)
print('{}'.format(bag(n,c,w,v)[n][c]))

 

  • 2
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值