QQ登录

只需要一步,快速开始

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

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

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

1189

主题

4

听众

2934

积分

该用户从未签到

跳转到指定楼层
1#
发表于 2023-11-7 11:17 |只看该作者 |倒序浏览
|招呼Ta 关注Ta
一、首先介绍一下0-1背包问题:  x# Z1 A$ M3 q8 m0 |) S
        0-1背包问题(0-1 Knapsack Problem)是一个经典的组合优化问题,通常在计算机科学和运筹学中讨论。这个问题涉及到一个背包和一组物品,每个物品都有一个特定的重量和价值。问题的目标是在给定背包的最大容量下,选择一组物品放入背包,以使得所选物品的总重量不超过背包的容量,同时最大化这些物品的总价值。
7 N4 e) s- J6 @+ @+ I+ ]5 a% i         0-1背包问题的名称中的“0-1”表示每个物品要么完全放入背包(选择)要么完全不放入背包(不选择),不能部分放入。这是问题的一个关键特征,与分数背包问题不同,分数背包问题允许部分放入物品。, b/ J# {2 e! A8 x% ]
问题的形式描述如下:

0 U9 u9 h# h1 S. j3 \$ j                 1.给定一个固定容量的背包,通常表示为一个正整数W(背包的最大承载重量)。
' D* g0 N* Y3 f6 {                 2.给定一组物品,每个物品都有两个属性:重量(weight)和价值(value)。. q% |( W- S" O8 i- S$ G& m+ m
                 3.对每个物品,你可以选择将其放入背包(选择)或不放入背包(不选择)。6 R( H0 l" ?! R% W) n3 x
                 4.每个物品只能选择一次,即要么放入背包,要么不放入。! |1 N. X6 q( b6 }# V
                 5.目标是选择一个物品组合,使得它们的总重量不超过背包容量W,同时使它们的总价值最大化。
  w# g# m8 C; `7 F( \2 U
" N. b. ?# u/ C  ^) B, k! H- f% d1 _解决0-1背包问题的一种常见方法是使用动态规划(Dynamic Programming)算法。这个问题有广泛的应用,包括资源分配、排程问题、投资组合优化等领域。它还是计算复杂性理论中的一个经典问题,通常被用来说明NP难问题的概念。& J: G1 d3 R+ X  w) g0 n
- j, N: O8 P% ^% ~. s. H: p6 ~

# X: G9 O2 `; h2 Q二、 介绍代码
4 ^' h* K+ d" S) g' X- e4 c- Y; w3 o这段代码是一个Python实现的0-1背包问题的解决方法,使用了动态规划算法来找到最优解。以下是对代码的详细解释:
  1. def knapsack(v, w, n, capacity):: D0 e* x! F- v9 l% h
  2.     i = 0
    \" o0 t9 b7 S. a
  3.     capacity = capacity + 1  # 初始化背包容量最大值2 C6 h5 s) G- t. H. s
  4.     m = np.zeros((n, capacity))  # 初始化
    8 B9 p1 J& d6 R% t\" E  N! {
  5.     x = np.zeros(n)
复制代码
1.v 是一个长度为 n 的列表,表示每个物品的价值。
/ b% `( r2 U, i/ m7 l2.w 是一个长度为 n 的列表,表示每个物品的重量。
& R: W, Z; l* a) _3.n 表示物品的数量。
  N1 t3 o  J  o0 Q4.capacity 表示背包的容量。7 t- N: f% E+ ?4 n6 B# \+ @, K

3 m* e0 w. M. s; u0 Y7 z. Z代码首先初始化了一个二维数组 m 作为动态规划表,其中 m[j] 表示在考虑前 i 个物品时,背包容量为 j 时可以获得的最大总价值。数组 x 用来存储最终的解,x 表示是否选择第 i 个物品。
  1.     for i in range(n):
    % U\" R9 Y6 s; s+ O
  2.         for j in range(capacity):) j* s7 b6 U1 |
  3.             if (j >= w[i]):
    * m0 R5 D\" v8 A' q
  4.                 m[i][j] = max(m[i - 1][j], m[i - 1][j - w[i]] + v[i])) L3 @0 ]; M, {, }\" q2 P
  5.             else:# R5 @% U# F( |( X
  6.                 m[i][j] = m[i - 1][j]
复制代码
在这个部分,代码使用了一个嵌套的循环,遍历了所有物品和不同的背包容量。对于每个物品 i 和容量 j,代码计算了两种情况下的最大总价值:& Z+ [" [1 H6 {, ^1 {! I) q  c1 k
5.如果当前物品的重量 w 小于等于当前容量 j,那么可以选择将第 i 个物品放入背包,此时总价值为 m[i-1][j-w] + v,或者选择不放入,此时总价值为 m[i-1][j]。代码选择其中较大的值作为 m[j]。6 m( m$ ^4 ^# p1 z
6.如果当前物品的重量 w 大于当前容量 j,则无法放入物品,所以总价值等于上一行的值 m[i-1][j]。  i1 y' A, D9 w1 C5 V
# F" D# v0 y0 O# X, `  h+ i4 ~
这个循环填充了动态规划表 m,最终 m[n-1][capacity] 包含了问题的最优解,即在给定容量下可以获得的最大总价值。
  1.     capacity = capacity - 1
    ; W! O# }7 U% N& u' m7 D# _* t
  2.     for i in range(n - 1, 0, -1):' o; H1 Q& Z8 _* t* C0 l
  3.         if (m[i][capacity] == m[i - 1][capacity]):
    # X: V. v! H4 N' {' k
  4.             x[i] = 03 @3 W; z+ I- u) Y
  5.         else:
    $ I+ M9 H% K! v4 J
  6.             x[i] = 1# i. p4 B/ G) ^# m\" ?: {+ S
  7.             capacity -= w[i]8 d0 `# }- d4 z- o
  8.     x[0] = 1 if (m[1][capacity] > 0) else 0
复制代码
在这一部分,代码反向遍历动态规划表,从最后一行向前找到解的路径。如果 m[capacity] 等于 m[i-1][capacity],表示第 i 个物品没有放入背包,否则放入背包,并更新剩余容量 capacity。
  1.     weight = 0* y7 W* Q2 k3 {  [6 n- k
  2.     value = 0, o4 ~7 A: e1 x% r$ e8 z( [6 |' u5 A
  3.     print('装载的物品编号为:')
    7 M! W2 @( l; }
  4.     for i in range(len(x)):
    \" v* @5 f3 u/ Y9 G( ]
  5.         if (x[i] == 1):
    2 Z6 K  }1 X4 {1 }
  6.             weight = weight + w[i]1 [1 ^1 t1 A. e2 u
  7.             value = value + v[i]
    , [  @0 `( J) p5 k
  8.             print(' ', i + 1)0 N7 f% `4 r0 i- |( l6 `
  9.     print('装载的物品重量为:')+ R9 @$ h; [' z  f  ~
  10.     print(weight)
    - w- k+ Q+ O- N+ Y* h4 Z$ T
  11.     print('装入的物品价值为:')
    0 M4 U* x: t8 C# _& E- ^' w4 U
  12.     print(value)
    4 T) k% a& V: y! o$ g
  13.     return m
复制代码
最后,代码计算了被选择的物品的总重量和总价值,并将它们打印出来。函数返回动态规划表 m。
: K& \2 F7 H  ^& G$ [5 C这段代码实现了0-1背包问题的解决方法,它通过动态规划算法找到最优解,即在给定背包容量下可以获得的最大总价值,以及选择哪些物品放入背包。
" E# }* w, F/ {  p& ]# t5 m3 a' S+ L- }7 ^2 n
最后在函数的输入文件中,如下例,第一行物品数量为:5 背包载重量为:10物品的重量列表为:[2, 2, 6, 5, 4] 物品的价值列表为: [6, 3, 5, 4, 6]4 W" @  G( I6 Y2 `" o; y% M2 g( m
VeryCapture_20231107110045.jpg

, B) V2 `7 Q8 L% l1 A3 q5 I4 Z) Z
, J/ n  k' m. \5 x; W
接下来展示我们的输出结果:
$ y8 ]0 ?9 o+ s9 `. F; p
VeryCapture_20231107110508.jpg
1 r) D$ @/ x5 c) A2 X) i5 T
, e7 Q; K$ O& W% F, _# |: A  @! \
# K2 R8 F" C& e% v/ B# W# a
具体代码如下:

! l1 q2 t+ a# C
6 R/ H- S) T7 H& x5 {
* ^7 E6 h: }+ i" M. ^- {4 U7 B6 D
* d  g2 a6 m7 g" L

; M5 z/ B* M8 E* U$ b4 p3 K% h: b" B" U. j2 z7 y1 A& G

基础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-6-5 03:28 , Processed in 1.048009 second(s), 55 queries .

回顶部