QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 2626|回复: 0
打印 上一主题 下一主题

python代码解决0-1背包问题

[复制链接]
字体大小: 正常 放大

1186

主题

4

听众

2922

积分

该用户从未签到

跳转到指定楼层
1#
发表于 2023-11-7 11:17 |只看该作者 |正序浏览
|招呼Ta 关注Ta
一、首先介绍一下0-1背包问题:
( X6 T7 m  Y0 g7 J+ ^! P
        0-1背包问题(0-1 Knapsack Problem)是一个经典的组合优化问题,通常在计算机科学和运筹学中讨论。这个问题涉及到一个背包和一组物品,每个物品都有一个特定的重量和价值。问题的目标是在给定背包的最大容量下,选择一组物品放入背包,以使得所选物品的总重量不超过背包的容量,同时最大化这些物品的总价值。& [" w  J& ~, N5 r& T4 g
         0-1背包问题的名称中的“0-1”表示每个物品要么完全放入背包(选择)要么完全不放入背包(不选择),不能部分放入。这是问题的一个关键特征,与分数背包问题不同,分数背包问题允许部分放入物品。+ c8 x4 O" U6 r* D% D3 Y7 W2 {
问题的形式描述如下:

7 ?$ J0 m3 E" a) ^& N+ T8 h$ `1 `                 1.给定一个固定容量的背包,通常表示为一个正整数W(背包的最大承载重量)。
7 r' t% {  y3 H; V7 D* _8 `                 2.给定一组物品,每个物品都有两个属性:重量(weight)和价值(value)。$ z! K3 ?- r1 [
                 3.对每个物品,你可以选择将其放入背包(选择)或不放入背包(不选择)。0 B% T  z$ e! c, H8 [3 @* r* g- C
                 4.每个物品只能选择一次,即要么放入背包,要么不放入。! y! S; Z6 h3 g' v% T( X0 V$ ^
                 5.目标是选择一个物品组合,使得它们的总重量不超过背包容量W,同时使它们的总价值最大化。2 o0 j  k) |4 [

4 q) |, I1 M& Q! i0 k: l解决0-1背包问题的一种常见方法是使用动态规划(Dynamic Programming)算法。这个问题有广泛的应用,包括资源分配、排程问题、投资组合优化等领域。它还是计算复杂性理论中的一个经典问题,通常被用来说明NP难问题的概念。8 A- g' a8 f0 m6 }, d$ e

3 R/ J7 g& d9 j, q6 A
; o4 `- ^3 P3 A二、 介绍代码
' u" n! F, d$ d: I% E$ r这段代码是一个Python实现的0-1背包问题的解决方法,使用了动态规划算法来找到最优解。以下是对代码的详细解释:
  1. def knapsack(v, w, n, capacity):
    1 g' h5 R6 x$ `
  2.     i = 0
    % M* ~% `' }& M& f; O
  3.     capacity = capacity + 1  # 初始化背包容量最大值
    3 \( m5 k' s# O1 }  u, H\" \( _
  4.     m = np.zeros((n, capacity))  # 初始化& `: g' o7 o7 f\" J+ Z! r8 i
  5.     x = np.zeros(n)
复制代码
1.v 是一个长度为 n 的列表,表示每个物品的价值。
( h7 d& G7 N) d9 h) d* x! L4 H2.w 是一个长度为 n 的列表,表示每个物品的重量。1 V! V+ Q3 E& K# u9 \, @: q3 c4 s
3.n 表示物品的数量。
3 u* P& b% P4 L7 F1 |, C' N4 I4.capacity 表示背包的容量。$ e2 ?  G+ Q/ B2 N
9 S+ d# j! q6 g
代码首先初始化了一个二维数组 m 作为动态规划表,其中 m[j] 表示在考虑前 i 个物品时,背包容量为 j 时可以获得的最大总价值。数组 x 用来存储最终的解,x 表示是否选择第 i 个物品。
  1.     for i in range(n):. d6 J9 I; s9 @
  2.         for j in range(capacity):
    . }' R5 g1 ^( D& \
  3.             if (j >= w[i]):
    9 I4 U2 E+ ^; I2 p- O3 u- B
  4.                 m[i][j] = max(m[i - 1][j], m[i - 1][j - w[i]] + v[i])
    ( z& _2 L4 ?& p, Z5 C
  5.             else:
    $ O7 ~  a9 o3 J. d: r' x
  6.                 m[i][j] = m[i - 1][j]
复制代码
在这个部分,代码使用了一个嵌套的循环,遍历了所有物品和不同的背包容量。对于每个物品 i 和容量 j,代码计算了两种情况下的最大总价值:
: g6 N( B- K& x9 z3 V. t5.如果当前物品的重量 w 小于等于当前容量 j,那么可以选择将第 i 个物品放入背包,此时总价值为 m[i-1][j-w] + v,或者选择不放入,此时总价值为 m[i-1][j]。代码选择其中较大的值作为 m[j]。
+ N) ?& S% E* F, o4 G6 z: e' n6.如果当前物品的重量 w 大于当前容量 j,则无法放入物品,所以总价值等于上一行的值 m[i-1][j]。3 F' A: G! m" t% a+ ?; n
* o6 S( g' Z6 y' I. y, s$ W9 I( y' H
这个循环填充了动态规划表 m,最终 m[n-1][capacity] 包含了问题的最优解,即在给定容量下可以获得的最大总价值。
  1.     capacity = capacity - 1) T7 A0 z1 i; D9 W& H
  2.     for i in range(n - 1, 0, -1):# ?/ `% U6 [- @
  3.         if (m[i][capacity] == m[i - 1][capacity]):6 Y  p3 {1 s9 m  m* N1 F. u1 _, I
  4.             x[i] = 0* s3 B: z1 E3 Y+ U
  5.         else:\" g6 h. |; k- @7 V
  6.             x[i] = 1+ Z) O7 d: c$ R7 X# }4 f0 H' i
  7.             capacity -= w[i]
    1 s/ r3 n/ q0 U% l; W+ s6 @- R4 O
  8.     x[0] = 1 if (m[1][capacity] > 0) else 0
复制代码
在这一部分,代码反向遍历动态规划表,从最后一行向前找到解的路径。如果 m[capacity] 等于 m[i-1][capacity],表示第 i 个物品没有放入背包,否则放入背包,并更新剩余容量 capacity。
  1.     weight = 0+ |# v8 T; @+ X3 Z* R
  2.     value = 0
    : O: R2 U0 g. {7 Y3 w; C
  3.     print('装载的物品编号为:')9 g+ m1 Z5 B0 ]  v6 j
  4.     for i in range(len(x)):5 R) ]8 k5 U% ^9 ~5 A7 L2 X$ M( @8 Z
  5.         if (x[i] == 1):
    , p8 z\" f$ O\" V' I. A
  6.             weight = weight + w[i]
    9 ~5 e) y) d& l/ ~& K' u: l2 p
  7.             value = value + v[i]1 \, j6 c1 q1 o\" F% u
  8.             print(' ', i + 1)7 @, X; @' B\" N6 o
  9.     print('装载的物品重量为:')
    & i% b0 K. j# q1 _( _9 k
  10.     print(weight)
    ' b' M; L; Q% p+ S5 X) W; m% O
  11.     print('装入的物品价值为:')
    2 C- }- j. s\" r- Q\" e
  12.     print(value)
    5 b\" J2 D. }\" H( @- X) N& D9 I
  13.     return m
复制代码
最后,代码计算了被选择的物品的总重量和总价值,并将它们打印出来。函数返回动态规划表 m。
; {1 D6 C( C( \/ Y# C/ J) f. @& _* c2 m这段代码实现了0-1背包问题的解决方法,它通过动态规划算法找到最优解,即在给定背包容量下可以获得的最大总价值,以及选择哪些物品放入背包。" }- x8 n5 g# c+ r9 f% R
* Y& W7 Z' r3 u6 i( t5 H) k2 ?
最后在函数的输入文件中,如下例,第一行物品数量为:5 背包载重量为:10物品的重量列表为:[2, 2, 6, 5, 4] 物品的价值列表为: [6, 3, 5, 4, 6]
' t+ L0 R( T6 M2 j9 {  P) }
VeryCapture_20231107110045.jpg

! D% [( F# _2 K% E) W3 h+ l, f" o- |# G% |0 O9 v
接下来展示我们的输出结果:
1 i/ d1 w) a0 ^) P; J2 b" B! t
VeryCapture_20231107110508.jpg
9 P  _1 m: S8 g1 [% x
. o/ y( R3 d  a7 P3 R2 d
+ |( b* P$ n( d! U
具体代码如下:
6 k* Y, f  R0 R( z/ r( ~' H

) \8 G) ^. p7 K" r" v3 E6 F6 L2 F+ ?' `% P  @+ k" b& @" ~& {( \. N0 {
4 G. o1 `+ S2 n8 j, r
, P& W1 A+ y- A/ _8 T

3 W+ x9 S, H& g3 w' Z+ ^9 a/ T

基础0-1背包问题(动态规划).rar

3.58 KB, 下载次数: 0, 下载积分: 体力 -2 点

售价: 2 点体力  [记录]  [购买]

zan
转播转播0 分享淘帖0 分享分享0 收藏收藏0 支持支持0 反对反对0 微信微信
您需要登录后才可以回帖 登录 | 注册地址

qq
收缩
  • 电话咨询

  • 04714969085
fastpost

关于我们| 联系我们| 诚征英才| 对外合作| 产品服务| QQ

手机版|Archiver| |繁體中文 手机客户端  

蒙公网安备 15010502000194号

Powered by Discuz! X2.5   © 2001-2013 数学建模网-数学中国 ( 蒙ICP备14002410号-3 蒙BBS备-0002号 )     论坛法律顾问:王兆丰

GMT+8, 2026-4-15 16:24 , Processed in 0.417314 second(s), 55 queries .

回顶部