QQ登录

只需要一步,快速开始

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

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

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

1186

主题

4

听众

2922

积分

该用户从未签到

跳转到指定楼层
1#
发表于 2023-11-7 11:17 |只看该作者 |倒序浏览
|招呼Ta 关注Ta
一、首先介绍一下0-1背包问题:; [1 x+ a( C7 v; u; g( @
        0-1背包问题(0-1 Knapsack Problem)是一个经典的组合优化问题,通常在计算机科学和运筹学中讨论。这个问题涉及到一个背包和一组物品,每个物品都有一个特定的重量和价值。问题的目标是在给定背包的最大容量下,选择一组物品放入背包,以使得所选物品的总重量不超过背包的容量,同时最大化这些物品的总价值。
' g  d2 R# v# {) S& T) R# y         0-1背包问题的名称中的“0-1”表示每个物品要么完全放入背包(选择)要么完全不放入背包(不选择),不能部分放入。这是问题的一个关键特征,与分数背包问题不同,分数背包问题允许部分放入物品。% Y  D5 Y9 `5 ^% `6 D9 F
问题的形式描述如下:
! |1 ?6 m& @. s% Y( s7 P
                 1.给定一个固定容量的背包,通常表示为一个正整数W(背包的最大承载重量)。7 P$ R& l1 Z& A" Y
                 2.给定一组物品,每个物品都有两个属性:重量(weight)和价值(value)。
& D/ K! H( V: E: b) u, z                 3.对每个物品,你可以选择将其放入背包(选择)或不放入背包(不选择)。* G* o% n/ D# f
                 4.每个物品只能选择一次,即要么放入背包,要么不放入。
$ j0 v' y0 i) N( C4 C8 f                 5.目标是选择一个物品组合,使得它们的总重量不超过背包容量W,同时使它们的总价值最大化。: b  v7 B. `7 j; n5 c
! u3 n5 f  x/ u7 }- ?
解决0-1背包问题的一种常见方法是使用动态规划(Dynamic Programming)算法。这个问题有广泛的应用,包括资源分配、排程问题、投资组合优化等领域。它还是计算复杂性理论中的一个经典问题,通常被用来说明NP难问题的概念。+ w& h8 y4 c1 E) x2 K6 a" G5 }3 i

) R( z. c& Z3 f0 J+ Q. x0 H; R. e  E4 c) R) ?! v  Y7 S
二、 介绍代码
; D/ e, B! I; e+ l! n6 h* a这段代码是一个Python实现的0-1背包问题的解决方法,使用了动态规划算法来找到最优解。以下是对代码的详细解释:
  1. def knapsack(v, w, n, capacity):  I& \8 V8 b( c. Z1 O7 |/ D4 }0 W
  2.     i = 02 T; C+ Y5 ~* Y1 ~  N
  3.     capacity = capacity + 1  # 初始化背包容量最大值9 [8 F1 ]' h8 N( `3 a& ^, C( B7 y
  4.     m = np.zeros((n, capacity))  # 初始化1 U\" b. D5 _/ ^* p4 j/ R0 [! r\" P1 R9 ~
  5.     x = np.zeros(n)
复制代码
1.v 是一个长度为 n 的列表,表示每个物品的价值。
9 g, U2 W! l" _6 i$ M2.w 是一个长度为 n 的列表,表示每个物品的重量。' W* k4 b7 p9 M! i9 w; h6 R# L
3.n 表示物品的数量。
( b$ D* Q% i9 H( a: N4.capacity 表示背包的容量。
0 ]9 a  m% E! L/ \' p, K5 n( x0 ^' s5 @
代码首先初始化了一个二维数组 m 作为动态规划表,其中 m[j] 表示在考虑前 i 个物品时,背包容量为 j 时可以获得的最大总价值。数组 x 用来存储最终的解,x 表示是否选择第 i 个物品。
  1.     for i in range(n):
    1 _% ]7 w6 }1 U& |9 M
  2.         for j in range(capacity):: v% I0 n: w0 o* g3 N2 B
  3.             if (j >= w[i]):
    * G3 C% y5 y' B2 S6 s
  4.                 m[i][j] = max(m[i - 1][j], m[i - 1][j - w[i]] + v[i]). k0 P: k; K6 N1 P  }
  5.             else:
    5 B0 L3 j, g& K
  6.                 m[i][j] = m[i - 1][j]
复制代码
在这个部分,代码使用了一个嵌套的循环,遍历了所有物品和不同的背包容量。对于每个物品 i 和容量 j,代码计算了两种情况下的最大总价值:; A- ^! G4 x5 J$ }0 o( N5 V( u' z
5.如果当前物品的重量 w 小于等于当前容量 j,那么可以选择将第 i 个物品放入背包,此时总价值为 m[i-1][j-w] + v,或者选择不放入,此时总价值为 m[i-1][j]。代码选择其中较大的值作为 m[j]。/ M  i* f% b' I3 v# k- Y6 }. P
6.如果当前物品的重量 w 大于当前容量 j,则无法放入物品,所以总价值等于上一行的值 m[i-1][j]。
  ~; X7 @/ R( _  D3 ?; \5 Y: E( w: J9 c5 _  D) c
这个循环填充了动态规划表 m,最终 m[n-1][capacity] 包含了问题的最优解,即在给定容量下可以获得的最大总价值。
  1.     capacity = capacity - 11 Q) Z' \- K/ j1 j& Q
  2.     for i in range(n - 1, 0, -1):
    , u+ U4 z7 @) _& I\" j
  3.         if (m[i][capacity] == m[i - 1][capacity]):9 T$ Y4 ~8 {' }) Q\" D4 V3 |
  4.             x[i] = 0
    ' p# a6 J( P) d3 L' x8 ]: B
  5.         else:
    , X# G! \. H) ?- B
  6.             x[i] = 1
    : _0 F\" y/ s9 e! I' M
  7.             capacity -= w[i]9 r' y1 e- ^8 p\" ~; \' J7 H
  8.     x[0] = 1 if (m[1][capacity] > 0) else 0
复制代码
在这一部分,代码反向遍历动态规划表,从最后一行向前找到解的路径。如果 m[capacity] 等于 m[i-1][capacity],表示第 i 个物品没有放入背包,否则放入背包,并更新剩余容量 capacity。
  1.     weight = 0
    % {% z( ~4 `8 j, D
  2.     value = 0$ S& v, w) q0 h# p\" `% n) o
  3.     print('装载的物品编号为:')$ F* Z\" v) e+ y$ P% V2 P
  4.     for i in range(len(x)):
    # C+ B+ m& ?3 S. x\" p, S
  5.         if (x[i] == 1):\" z0 D! w6 @2 ^& N- X
  6.             weight = weight + w[i]
    + J/ \& T9 d' C& ]7 M$ O
  7.             value = value + v[i]
    : m: G( ]: r5 G: A* x* L. |1 ]% I
  8.             print(' ', i + 1)
    $ C9 n\" e% m6 l; c3 k6 c+ S
  9.     print('装载的物品重量为:')
    $ p1 y1 n( k0 g: h8 w+ n' L7 |
  10.     print(weight)+ g3 R) \9 P/ n\" s- C3 z
  11.     print('装入的物品价值为:')
    3 R* F4 F3 A( p  l3 X$ T
  12.     print(value)
    - T9 f4 P. C2 C, l& H. u+ O, D
  13.     return m
复制代码
最后,代码计算了被选择的物品的总重量和总价值,并将它们打印出来。函数返回动态规划表 m。0 t! a7 h( C4 U
这段代码实现了0-1背包问题的解决方法,它通过动态规划算法找到最优解,即在给定背包容量下可以获得的最大总价值,以及选择哪些物品放入背包。  a" n  O8 t( M5 S$ `

' y( p# [5 \/ c7 K最后在函数的输入文件中,如下例,第一行物品数量为:5 背包载重量为:10物品的重量列表为:[2, 2, 6, 5, 4] 物品的价值列表为: [6, 3, 5, 4, 6]' B- u2 ]; N: Q9 r, j
VeryCapture_20231107110045.jpg

0 [: L/ [  G" T, ~" T
% s3 ~, o; R# T: L, f
接下来展示我们的输出结果:
+ C( w$ W7 A" t
VeryCapture_20231107110508.jpg
" ?5 M7 z0 e1 D7 i
$ \* _! [( G  m7 F$ j

2 B& Z( F: P- F- P
具体代码如下:

5 V$ D- ?* f7 T( O4 d  ^; X, v) y  n: H7 F) d( o& V% A
+ z/ @0 T8 V! h/ x
" e/ ?2 _' Z6 u) J$ p/ y! R

2 A9 i2 y: H) w: ~% k0 O6 J" U6 w
, g( ?9 s6 `: g! l$ X

基础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 06:57 , Processed in 0.449096 second(s), 55 queries .

回顶部