QQ登录

只需要一步,快速开始

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

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

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

1186

主题

4

听众

2922

积分

该用户从未签到

跳转到指定楼层
1#
发表于 2023-11-7 11:17 |只看该作者 |倒序浏览
|招呼Ta 关注Ta
一、首先介绍一下0-1背包问题:
& s5 c2 E, W: x9 V2 g
        0-1背包问题(0-1 Knapsack Problem)是一个经典的组合优化问题,通常在计算机科学和运筹学中讨论。这个问题涉及到一个背包和一组物品,每个物品都有一个特定的重量和价值。问题的目标是在给定背包的最大容量下,选择一组物品放入背包,以使得所选物品的总重量不超过背包的容量,同时最大化这些物品的总价值。
8 P/ b$ u% c0 Z         0-1背包问题的名称中的“0-1”表示每个物品要么完全放入背包(选择)要么完全不放入背包(不选择),不能部分放入。这是问题的一个关键特征,与分数背包问题不同,分数背包问题允许部分放入物品。
5 r+ N/ `7 ?5 Z5 d) K# L% N- w
问题的形式描述如下:
5 I! s+ c$ G9 I7 _; Z
                 1.给定一个固定容量的背包,通常表示为一个正整数W(背包的最大承载重量)。
1 P9 p$ X/ B- k+ f1 L* r( t3 {                 2.给定一组物品,每个物品都有两个属性:重量(weight)和价值(value)。
4 C1 H! [, X5 O0 j8 o: l% e                 3.对每个物品,你可以选择将其放入背包(选择)或不放入背包(不选择)。
' ]  P1 N3 V8 o/ y+ i                 4.每个物品只能选择一次,即要么放入背包,要么不放入。2 q& D7 k* C6 N. H& U8 M- A, F! Y. ^
                 5.目标是选择一个物品组合,使得它们的总重量不超过背包容量W,同时使它们的总价值最大化。
0 }# J. W  b: `7 L: Q2 b9 J- V7 K7 H% L. U; V8 D9 V
解决0-1背包问题的一种常见方法是使用动态规划(Dynamic Programming)算法。这个问题有广泛的应用,包括资源分配、排程问题、投资组合优化等领域。它还是计算复杂性理论中的一个经典问题,通常被用来说明NP难问题的概念。, ^- h  ]9 N3 O+ {/ ^
. `% Y/ s, |# b- z' c' q2 h

$ |! ]6 L+ B* K$ l3 K* L' \二、 介绍代码& ]. H( f! N& @2 {
这段代码是一个Python实现的0-1背包问题的解决方法,使用了动态规划算法来找到最优解。以下是对代码的详细解释:
  1. def knapsack(v, w, n, capacity):& _. o& N/ C2 [7 R% R
  2.     i = 0
    : r$ Z2 o\" z% z/ \$ z' z
  3.     capacity = capacity + 1  # 初始化背包容量最大值
    % j* M9 @6 S% u0 V+ H5 H$ y
  4.     m = np.zeros((n, capacity))  # 初始化
    ' m1 W7 N  v# {, X1 s% @3 @7 H- r1 b
  5.     x = np.zeros(n)
复制代码
1.v 是一个长度为 n 的列表,表示每个物品的价值。# ]8 W& @) J0 x2 O+ b; J& Q# }+ [
2.w 是一个长度为 n 的列表,表示每个物品的重量。
* \6 p8 a# E7 n' ]1 U3.n 表示物品的数量。
' `& e* [! M; V) t7 p4.capacity 表示背包的容量。
  p" q! T" G; V0 }0 f0 X8 r1 A* F( x0 l# t4 C. X
代码首先初始化了一个二维数组 m 作为动态规划表,其中 m[j] 表示在考虑前 i 个物品时,背包容量为 j 时可以获得的最大总价值。数组 x 用来存储最终的解,x 表示是否选择第 i 个物品。
  1.     for i in range(n):
    ) t* E2 p8 Y$ W4 {\" L7 O( m
  2.         for j in range(capacity):
    / h, K2 ?& R% `  X$ v$ y& L* |9 g
  3.             if (j >= w[i]):% C2 N2 D3 F; k
  4.                 m[i][j] = max(m[i - 1][j], m[i - 1][j - w[i]] + v[i])
    1 O6 q5 [1 o& C) ?! @, a/ c7 M
  5.             else:+ [) l) ~' y5 }5 Y! {; |
  6.                 m[i][j] = m[i - 1][j]
复制代码
在这个部分,代码使用了一个嵌套的循环,遍历了所有物品和不同的背包容量。对于每个物品 i 和容量 j,代码计算了两种情况下的最大总价值:( i7 a  v4 F  o3 C' Q$ }2 q
5.如果当前物品的重量 w 小于等于当前容量 j,那么可以选择将第 i 个物品放入背包,此时总价值为 m[i-1][j-w] + v,或者选择不放入,此时总价值为 m[i-1][j]。代码选择其中较大的值作为 m[j]。
7 j1 }+ c: E& N; u3 s0 r% @) w6.如果当前物品的重量 w 大于当前容量 j,则无法放入物品,所以总价值等于上一行的值 m[i-1][j]。# d+ }4 I2 Y" s( M7 e
' e2 y: N8 Z' e
这个循环填充了动态规划表 m,最终 m[n-1][capacity] 包含了问题的最优解,即在给定容量下可以获得的最大总价值。
  1.     capacity = capacity - 1
    8 r\" h% L5 W* W% M  R5 D- Q1 X
  2.     for i in range(n - 1, 0, -1):
    % l( t* M9 C* G* r2 ?# F% U# Y
  3.         if (m[i][capacity] == m[i - 1][capacity]):
    ( E7 _1 Q* V3 Z, G* p: }' v* c
  4.             x[i] = 04 F0 J; l6 P5 o9 @* ^# D
  5.         else:$ s8 T7 }2 ?* u0 F$ v* ]
  6.             x[i] = 1% Y9 ?, t# B8 O/ v& W3 e6 a! |. z; F
  7.             capacity -= w[i]: q+ f  ^7 F! J) T2 a
  8.     x[0] = 1 if (m[1][capacity] > 0) else 0
复制代码
在这一部分,代码反向遍历动态规划表,从最后一行向前找到解的路径。如果 m[capacity] 等于 m[i-1][capacity],表示第 i 个物品没有放入背包,否则放入背包,并更新剩余容量 capacity。
  1.     weight = 0
    ! _3 v' h- s8 Z3 T0 e) w
  2.     value = 0
    , |# H! D3 i/ J' f  R1 s
  3.     print('装载的物品编号为:')1 J. y' z- [* D! L; [* ~3 c0 J' G
  4.     for i in range(len(x)):8 ^! |8 ?& `1 R9 O$ R
  5.         if (x[i] == 1):$ ?( o8 O3 z- A9 Y8 p
  6.             weight = weight + w[i]0 w, W9 ~3 s( k% M& W+ M& A
  7.             value = value + v[i]
    % G9 {9 d0 u$ l\" O& v* i; X* b5 X) ^
  8.             print(' ', i + 1)3 m- ^/ x; [: T$ k
  9.     print('装载的物品重量为:')
    ( _! k5 p! v! D- x# g5 t4 D# ^
  10.     print(weight)5 l- e& D8 l+ O6 j, z
  11.     print('装入的物品价值为:')
    1 Q0 g$ }2 W' ]
  12.     print(value)* D  @4 y. ]% w% g$ W
  13.     return m
复制代码
最后,代码计算了被选择的物品的总重量和总价值,并将它们打印出来。函数返回动态规划表 m。& `8 F, ?* a* S" ]
这段代码实现了0-1背包问题的解决方法,它通过动态规划算法找到最优解,即在给定背包容量下可以获得的最大总价值,以及选择哪些物品放入背包。
# U+ R$ h- I1 E, B
9 t0 k' s8 U6 x$ J最后在函数的输入文件中,如下例,第一行物品数量为:5 背包载重量为:10物品的重量列表为:[2, 2, 6, 5, 4] 物品的价值列表为: [6, 3, 5, 4, 6]
1 D' P( R+ p) E( \$ {
VeryCapture_20231107110045.jpg

% H* o, v( b% b% o* J( V& i% J) {6 k5 p6 U
接下来展示我们的输出结果:
5 |: M" Z9 @& g( @- b. M
VeryCapture_20231107110508.jpg

0 l, Q2 Y  G" ]

* _; D, v# R' V% U' {  J: H, E
  Q5 ?) L4 @" ~% F+ }  D
具体代码如下:
1 H) n9 R2 q  Q+ N+ r1 s
( G% W1 ]( u# E
& D) W* R+ f9 @: B
3 `0 m, n( p% m; M+ N2 O9 r. w

4 |& @& @) v# s2 Z" G9 |: @7 {2 o3 r% q

基础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 18:30 , Processed in 0.445766 second(s), 55 queries .

回顶部