QQ登录

只需要一步,快速开始

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

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

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

1188

主题

4

听众

2931

积分

该用户从未签到

跳转到指定楼层
1#
发表于 2023-11-7 11:17 |只看该作者 |倒序浏览
|招呼Ta 关注Ta
一、首先介绍一下0-1背包问题:
- ~/ [+ b, k1 }- H- ~
        0-1背包问题(0-1 Knapsack Problem)是一个经典的组合优化问题,通常在计算机科学和运筹学中讨论。这个问题涉及到一个背包和一组物品,每个物品都有一个特定的重量和价值。问题的目标是在给定背包的最大容量下,选择一组物品放入背包,以使得所选物品的总重量不超过背包的容量,同时最大化这些物品的总价值。0 b& L0 o( J* ~' J8 m. F
         0-1背包问题的名称中的“0-1”表示每个物品要么完全放入背包(选择)要么完全不放入背包(不选择),不能部分放入。这是问题的一个关键特征,与分数背包问题不同,分数背包问题允许部分放入物品。1 @3 T# A: I% P1 Q" f  G  x  }- I8 {  I
问题的形式描述如下:
# r+ Z5 F( @) R$ H, z
                 1.给定一个固定容量的背包,通常表示为一个正整数W(背包的最大承载重量)。! Y7 t% D6 D/ ^1 B
                 2.给定一组物品,每个物品都有两个属性:重量(weight)和价值(value)。" t% Q! g# e( u8 P" K/ G9 m
                 3.对每个物品,你可以选择将其放入背包(选择)或不放入背包(不选择)。
& s- k# e" Z! t                 4.每个物品只能选择一次,即要么放入背包,要么不放入。
# x1 q9 D8 ~  P$ r1 ]                 5.目标是选择一个物品组合,使得它们的总重量不超过背包容量W,同时使它们的总价值最大化。
; Y4 l# h  n2 n& a+ [' s
2 l/ z& n1 j' c  f$ J, V解决0-1背包问题的一种常见方法是使用动态规划(Dynamic Programming)算法。这个问题有广泛的应用,包括资源分配、排程问题、投资组合优化等领域。它还是计算复杂性理论中的一个经典问题,通常被用来说明NP难问题的概念。; o& F! M1 h9 V( D* P" n

9 C2 o+ o0 s6 E9 ]6 `7 L2 }" o3 i; {) A
二、 介绍代码
+ t3 ?! t7 ]6 q$ e这段代码是一个Python实现的0-1背包问题的解决方法,使用了动态规划算法来找到最优解。以下是对代码的详细解释:
  1. def knapsack(v, w, n, capacity):
    % u: J$ w8 d: s- r/ q, G* }
  2.     i = 0. }\" L4 ?  |  b9 \% V5 B
  3.     capacity = capacity + 1  # 初始化背包容量最大值& G$ h1 v; ^# y. d+ }
  4.     m = np.zeros((n, capacity))  # 初始化
    1 p: i( M* v: T\" q4 ^
  5.     x = np.zeros(n)
复制代码
1.v 是一个长度为 n 的列表,表示每个物品的价值。
4 _5 g4 o- l& S" p. m2 j( l( \2.w 是一个长度为 n 的列表,表示每个物品的重量。
. D. u& M: y' I: P3.n 表示物品的数量。
0 P/ U' x: Y+ K: _4.capacity 表示背包的容量。
& @0 Q: x6 ^/ P0 g5 t* L- g, M9 z- ^' E& _- u) Q( I
代码首先初始化了一个二维数组 m 作为动态规划表,其中 m[j] 表示在考虑前 i 个物品时,背包容量为 j 时可以获得的最大总价值。数组 x 用来存储最终的解,x 表示是否选择第 i 个物品。
  1.     for i in range(n):
    . @* i( t3 ^% ~
  2.         for j in range(capacity):
      |) o3 P: Z\" [& r3 F% b
  3.             if (j >= w[i]):! W% W% G7 M0 |9 `5 v; Q  g
  4.                 m[i][j] = max(m[i - 1][j], m[i - 1][j - w[i]] + v[i])
    ( E4 f% H9 ?4 R- B. X
  5.             else:7 E/ I1 I% O) N6 C* h6 F$ a
  6.                 m[i][j] = m[i - 1][j]
复制代码
在这个部分,代码使用了一个嵌套的循环,遍历了所有物品和不同的背包容量。对于每个物品 i 和容量 j,代码计算了两种情况下的最大总价值:1 U; R/ Z1 \* Z4 C, y% o5 f
5.如果当前物品的重量 w 小于等于当前容量 j,那么可以选择将第 i 个物品放入背包,此时总价值为 m[i-1][j-w] + v,或者选择不放入,此时总价值为 m[i-1][j]。代码选择其中较大的值作为 m[j]。# K  h. d" A" D# Q- L
6.如果当前物品的重量 w 大于当前容量 j,则无法放入物品,所以总价值等于上一行的值 m[i-1][j]。
+ H. }0 {4 c1 X9 L4 }& y, r. l8 o& z1 O- F# N) O9 s1 K
这个循环填充了动态规划表 m,最终 m[n-1][capacity] 包含了问题的最优解,即在给定容量下可以获得的最大总价值。
  1.     capacity = capacity - 1
    # i% U. C. U0 p! Q1 T1 H
  2.     for i in range(n - 1, 0, -1):
    : \, f& o, X4 S' x9 v
  3.         if (m[i][capacity] == m[i - 1][capacity]):
    ) e+ W4 {; s5 z; M# `( F+ f, c. R
  4.             x[i] = 0
    : j9 V\" ^\" I, n0 ?0 |, X$ d7 k8 [
  5.         else:
    * `2 @, U9 {+ v1 ?5 Z; b; t
  6.             x[i] = 1
    $ ?3 c) _; u5 T5 j
  7.             capacity -= w[i]
    ! w2 t9 j! I0 }% m+ x& z. g
  8.     x[0] = 1 if (m[1][capacity] > 0) else 0
复制代码
在这一部分,代码反向遍历动态规划表,从最后一行向前找到解的路径。如果 m[capacity] 等于 m[i-1][capacity],表示第 i 个物品没有放入背包,否则放入背包,并更新剩余容量 capacity。
  1.     weight = 0
    3 W% e, f\" w* [+ @
  2.     value = 09 ?/ Z' `7 `4 W5 Z) h
  3.     print('装载的物品编号为:')+ b  r, n% Q, T
  4.     for i in range(len(x)):
    \" Q, R; h$ ^. r- T
  5.         if (x[i] == 1):4 m\" M- U: U! j
  6.             weight = weight + w[i]5 ?+ B) Z1 M6 B( ~& o$ g
  7.             value = value + v[i]. |\" m0 F7 }. A. N5 w% F* x
  8.             print(' ', i + 1)* E( N8 y# n/ U& V5 _5 }8 O* n  Z\" Z
  9.     print('装载的物品重量为:')( n9 c0 N* q& B3 ~\" Q$ w
  10.     print(weight)3 `6 e- @; N, F
  11.     print('装入的物品价值为:')
    . N7 c3 U6 l  G: B1 G& H
  12.     print(value)
    ) V$ R6 y& ~\" h: F
  13.     return m
复制代码
最后,代码计算了被选择的物品的总重量和总价值,并将它们打印出来。函数返回动态规划表 m。
* @# e& Q* f) b: L( q/ M这段代码实现了0-1背包问题的解决方法,它通过动态规划算法找到最优解,即在给定背包容量下可以获得的最大总价值,以及选择哪些物品放入背包。
/ {8 s4 t: }; F2 h- G3 V: P$ t  i6 M7 b
最后在函数的输入文件中,如下例,第一行物品数量为:5 背包载重量为:10物品的重量列表为:[2, 2, 6, 5, 4] 物品的价值列表为: [6, 3, 5, 4, 6]
1 M  J1 s# t+ K* P) ^6 j3 @, _/ o
VeryCapture_20231107110045.jpg
) J+ w! Z+ J- A" o( S7 n/ v/ p' Z- w
+ [0 o; V  F5 ^# f
接下来展示我们的输出结果:

2 N+ g* w+ r% U9 \
VeryCapture_20231107110508.jpg

0 E9 W9 K* @" l7 n; B# b
' ~4 P5 O4 `. M: D' j  D6 W
2 t& R8 f: t) }! \% z
具体代码如下:
' _/ z$ U- P; k) X# R: f7 W
* ^) |8 s: M, q' [8 F
5 e- Q8 t' y2 U
- Q$ |1 B, [+ T* Q$ V# D5 I* Q
) N% w+ L8 J3 i

# m, d; V5 q, H9 e; d

基础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-5-25 18:06 , Processed in 0.423217 second(s), 55 queries .

回顶部