QQ登录

只需要一步,快速开始

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

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

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

1189

主题

4

听众

2934

积分

该用户从未签到

跳转到指定楼层
1#
发表于 2023-11-7 11:17 |只看该作者 |正序浏览
|招呼Ta 关注Ta
一、首先介绍一下0-1背包问题:
0 k0 Y+ a2 x1 H$ X, r) E$ u
        0-1背包问题(0-1 Knapsack Problem)是一个经典的组合优化问题,通常在计算机科学和运筹学中讨论。这个问题涉及到一个背包和一组物品,每个物品都有一个特定的重量和价值。问题的目标是在给定背包的最大容量下,选择一组物品放入背包,以使得所选物品的总重量不超过背包的容量,同时最大化这些物品的总价值。  W' ]) q1 L. d7 i& k8 ~% b
         0-1背包问题的名称中的“0-1”表示每个物品要么完全放入背包(选择)要么完全不放入背包(不选择),不能部分放入。这是问题的一个关键特征,与分数背包问题不同,分数背包问题允许部分放入物品。
+ S- l) ?. g* h
问题的形式描述如下:

( }& @( o2 f' ~* E0 ~6 h+ J  {7 k                 1.给定一个固定容量的背包,通常表示为一个正整数W(背包的最大承载重量)。, W0 r" ^8 V* q: i/ W# e* b9 s" @/ }
                 2.给定一组物品,每个物品都有两个属性:重量(weight)和价值(value)。
/ _) O  X, P" S) y                 3.对每个物品,你可以选择将其放入背包(选择)或不放入背包(不选择)。
8 L- C/ Z: s0 d# P- p                 4.每个物品只能选择一次,即要么放入背包,要么不放入。
+ Q$ L9 `4 T* D8 G0 K                 5.目标是选择一个物品组合,使得它们的总重量不超过背包容量W,同时使它们的总价值最大化。* N0 O: M+ n# l: ^, u  j4 [

" r, a4 Y- D5 F! Z: [+ r: |9 w, s解决0-1背包问题的一种常见方法是使用动态规划(Dynamic Programming)算法。这个问题有广泛的应用,包括资源分配、排程问题、投资组合优化等领域。它还是计算复杂性理论中的一个经典问题,通常被用来说明NP难问题的概念。+ [& T  ?) `$ x+ Z4 y. L& b
2 c8 J% N  m8 B3 J' P5 k

1 J7 I& \" f' t7 S二、 介绍代码
! m! C' |1 E1 x5 F( M* }这段代码是一个Python实现的0-1背包问题的解决方法,使用了动态规划算法来找到最优解。以下是对代码的详细解释:
  1. def knapsack(v, w, n, capacity):. _% i3 G2 b8 G2 m( _- f
  2.     i = 0' o- t0 q9 O' g& @6 C
  3.     capacity = capacity + 1  # 初始化背包容量最大值
    * M- e  u) Y' o$ v- _/ A  O% r& t\" e
  4.     m = np.zeros((n, capacity))  # 初始化
    - }7 D! y2 a\" f9 {, x- C, H
  5.     x = np.zeros(n)
复制代码
1.v 是一个长度为 n 的列表,表示每个物品的价值。8 @  A+ [8 y8 u9 o) M3 U; I
2.w 是一个长度为 n 的列表,表示每个物品的重量。
0 Y5 z. j0 T% z3 T& E3.n 表示物品的数量。0 P4 l9 X* q6 m
4.capacity 表示背包的容量。2 a; J* D8 I) ^) x; @

3 O7 Z5 r2 N' z- X1 j代码首先初始化了一个二维数组 m 作为动态规划表,其中 m[j] 表示在考虑前 i 个物品时,背包容量为 j 时可以获得的最大总价值。数组 x 用来存储最终的解,x 表示是否选择第 i 个物品。
  1.     for i in range(n):
    ! U. _' `' v: i* O\" I9 \
  2.         for j in range(capacity):
    6 ^; `  m7 a2 j! N# F. Z
  3.             if (j >= w[i]):) _  @' G6 [; z- d: a\" J
  4.                 m[i][j] = max(m[i - 1][j], m[i - 1][j - w[i]] + v[i])+ g# d) k/ Y7 ]) O8 E( p, `- B: }
  5.             else:
    4 t' e! K6 P5 x5 S! O. x. f! ]; T
  6.                 m[i][j] = m[i - 1][j]
复制代码
在这个部分,代码使用了一个嵌套的循环,遍历了所有物品和不同的背包容量。对于每个物品 i 和容量 j,代码计算了两种情况下的最大总价值:$ y! x+ v% s% o
5.如果当前物品的重量 w 小于等于当前容量 j,那么可以选择将第 i 个物品放入背包,此时总价值为 m[i-1][j-w] + v,或者选择不放入,此时总价值为 m[i-1][j]。代码选择其中较大的值作为 m[j]。
7 {# S1 y; X& A6 r6.如果当前物品的重量 w 大于当前容量 j,则无法放入物品,所以总价值等于上一行的值 m[i-1][j]。! Y+ _  z7 u+ U: V

% g0 {6 u2 u4 y& z- A6 J, ?+ c: k这个循环填充了动态规划表 m,最终 m[n-1][capacity] 包含了问题的最优解,即在给定容量下可以获得的最大总价值。
  1.     capacity = capacity - 1
    & J/ }+ Q, |! T9 S- H1 X0 P/ j
  2.     for i in range(n - 1, 0, -1):
    $ s2 I7 ]  s% ?\" F) _
  3.         if (m[i][capacity] == m[i - 1][capacity]):- Y6 ]0 i9 r' v6 d' B6 N8 l& v$ Q
  4.             x[i] = 0
    : g- r3 w: C! r2 f& s7 U9 {
  5.         else:  \. Y/ [7 @) y
  6.             x[i] = 1
    , U3 |  \3 ]' l+ X$ f, o
  7.             capacity -= w[i]) L8 l* F6 P; j\" T, O& I& e, R
  8.     x[0] = 1 if (m[1][capacity] > 0) else 0
复制代码
在这一部分,代码反向遍历动态规划表,从最后一行向前找到解的路径。如果 m[capacity] 等于 m[i-1][capacity],表示第 i 个物品没有放入背包,否则放入背包,并更新剩余容量 capacity。
  1.     weight = 09 }  P& A\" ]+ V3 X& e
  2.     value = 0\" c' t, L; i, n& [: c
  3.     print('装载的物品编号为:')
    ) x  p6 P) [5 C6 U\" V( I$ |
  4.     for i in range(len(x)):
    $ c' `# j) s) A! q* i  d$ O  k8 @
  5.         if (x[i] == 1):( |$ j' k) J1 ?0 h
  6.             weight = weight + w[i]1 R0 N( N5 L/ j) S3 w3 Z& f& ^
  7.             value = value + v[i]
    & C8 w; G- J2 t0 |
  8.             print(' ', i + 1)4 Q8 `# a; o% w8 ~0 r; q: R
  9.     print('装载的物品重量为:')3 T' D\" g$ C# N+ s6 ]
  10.     print(weight)
    ' j: a7 g# R: `. Z0 U: T
  11.     print('装入的物品价值为:')
    / A7 Q2 P6 K$ |# z- w( l
  12.     print(value)
    8 T1 }6 `- x- x# l. O6 ]3 ^! L, `
  13.     return m
复制代码
最后,代码计算了被选择的物品的总重量和总价值,并将它们打印出来。函数返回动态规划表 m。3 R" @  f8 E, Y  o/ M3 @
这段代码实现了0-1背包问题的解决方法,它通过动态规划算法找到最优解,即在给定背包容量下可以获得的最大总价值,以及选择哪些物品放入背包。: _% E: {: ^: r5 {

8 _- V7 J$ d4 B) ^. L0 X最后在函数的输入文件中,如下例,第一行物品数量为:5 背包载重量为:10物品的重量列表为:[2, 2, 6, 5, 4] 物品的价值列表为: [6, 3, 5, 4, 6]
+ J( b. r5 c( O/ m6 P$ f8 V& q- ~
VeryCapture_20231107110045.jpg
- B, _2 Z2 U+ S4 f8 h
& J' L" a- |! ^8 N9 ]
接下来展示我们的输出结果:

: z$ E+ `" |- U+ G2 K
VeryCapture_20231107110508.jpg

. R8 j; t( u' ?6 ^

; e  p+ {' Z6 c; N  Q
3 H( [4 S  B" w+ ]
具体代码如下:
2 |7 D; ^/ Q3 k% @- y( S

* g& E0 x# M+ W$ u% i$ P' P
! N& z" ^' I/ {8 r% q
( M! j2 U  z& N  Y. U

% k% J( \2 `' z$ I( Y0 \
6 z- @: B* I# a& x' \' 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-6-5 07:26 , Processed in 2.343463 second(s), 55 queries .

回顶部