QQ登录

只需要一步,快速开始

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

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

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

1186

主题

4

听众

2922

积分

该用户从未签到

跳转到指定楼层
1#
发表于 2023-11-7 11:17 |只看该作者 |倒序浏览
|招呼Ta 关注Ta
一、首先介绍一下0-1背包问题:( y5 z/ S) A& H) \) A. p8 D
        0-1背包问题(0-1 Knapsack Problem)是一个经典的组合优化问题,通常在计算机科学和运筹学中讨论。这个问题涉及到一个背包和一组物品,每个物品都有一个特定的重量和价值。问题的目标是在给定背包的最大容量下,选择一组物品放入背包,以使得所选物品的总重量不超过背包的容量,同时最大化这些物品的总价值。" N$ j: c: m. K& T+ q& P
         0-1背包问题的名称中的“0-1”表示每个物品要么完全放入背包(选择)要么完全不放入背包(不选择),不能部分放入。这是问题的一个关键特征,与分数背包问题不同,分数背包问题允许部分放入物品。
  Z7 U5 j" Q! \
问题的形式描述如下:
# m8 b# M* l  l
                 1.给定一个固定容量的背包,通常表示为一个正整数W(背包的最大承载重量)。
6 Y" P" y0 Y5 C9 ?* D                 2.给定一组物品,每个物品都有两个属性:重量(weight)和价值(value)。
; E% _! K8 R) V0 C" Q, K4 T9 T                 3.对每个物品,你可以选择将其放入背包(选择)或不放入背包(不选择)。, S9 h) s9 N& E% i% m& w7 z* m
                 4.每个物品只能选择一次,即要么放入背包,要么不放入。) `* J) ^1 W% O5 ?3 l
                 5.目标是选择一个物品组合,使得它们的总重量不超过背包容量W,同时使它们的总价值最大化。
" [3 V) i- s2 N
  e" A  f# L$ v+ p, C; b5 a解决0-1背包问题的一种常见方法是使用动态规划(Dynamic Programming)算法。这个问题有广泛的应用,包括资源分配、排程问题、投资组合优化等领域。它还是计算复杂性理论中的一个经典问题,通常被用来说明NP难问题的概念。6 G9 Z1 A9 e; [% c$ I( X

, G% R$ O% \- p8 l& j4 {9 H* P& |# K
二、 介绍代码; A! z/ z' p* a/ |! Q( Y( i
这段代码是一个Python实现的0-1背包问题的解决方法,使用了动态规划算法来找到最优解。以下是对代码的详细解释:
  1. def knapsack(v, w, n, capacity):* A' Z/ b7 x; R, n  X
  2.     i = 0! u8 F  N2 Q& U3 c5 W
  3.     capacity = capacity + 1  # 初始化背包容量最大值
    ) L  k% W3 i% x; c0 I
  4.     m = np.zeros((n, capacity))  # 初始化' N: S4 q% G\" H
  5.     x = np.zeros(n)
复制代码
1.v 是一个长度为 n 的列表,表示每个物品的价值。
9 r9 D. |+ \2 A; x$ O0 ^2.w 是一个长度为 n 的列表,表示每个物品的重量。! }  B3 h1 T1 Y( P2 B) a
3.n 表示物品的数量。
3 |2 _  V3 t! j4.capacity 表示背包的容量。
4 a, ]& P, m/ K9 k' s! _
6 n4 r: ~8 z, w" V: p( K代码首先初始化了一个二维数组 m 作为动态规划表,其中 m[j] 表示在考虑前 i 个物品时,背包容量为 j 时可以获得的最大总价值。数组 x 用来存储最终的解,x 表示是否选择第 i 个物品。
  1.     for i in range(n):
    ) G7 o4 {3 a. C\" W* u: ^. w
  2.         for j in range(capacity):1 J, s) O& ?0 Q# r: P6 A+ L
  3.             if (j >= w[i]):
    , U) }2 E) L* y- t
  4.                 m[i][j] = max(m[i - 1][j], m[i - 1][j - w[i]] + v[i])/ }, i7 y5 q# C% P: [
  5.             else:
    1 D  {% L/ m) B8 C& A
  6.                 m[i][j] = m[i - 1][j]
复制代码
在这个部分,代码使用了一个嵌套的循环,遍历了所有物品和不同的背包容量。对于每个物品 i 和容量 j,代码计算了两种情况下的最大总价值:$ M8 h' U$ ?2 G
5.如果当前物品的重量 w 小于等于当前容量 j,那么可以选择将第 i 个物品放入背包,此时总价值为 m[i-1][j-w] + v,或者选择不放入,此时总价值为 m[i-1][j]。代码选择其中较大的值作为 m[j]。
6 p( s" P/ A& ]8 e  r% d2 r1 \5 F) Z' \6.如果当前物品的重量 w 大于当前容量 j,则无法放入物品,所以总价值等于上一行的值 m[i-1][j]。
; Y2 `" C5 N) R9 h
' y8 @4 [  c! i这个循环填充了动态规划表 m,最终 m[n-1][capacity] 包含了问题的最优解,即在给定容量下可以获得的最大总价值。
  1.     capacity = capacity - 1( Q, ]# j+ ^1 j. G( K: p
  2.     for i in range(n - 1, 0, -1):
    7 O0 ~8 h\" j  b9 |7 x
  3.         if (m[i][capacity] == m[i - 1][capacity]):
    $ C6 ]0 k- o9 i6 e
  4.             x[i] = 00 u7 d! H  q, }' p2 n& {- u
  5.         else:
    % X% W- p; |$ w1 S! b) q
  6.             x[i] = 1# R\" S! l7 c0 m) T4 U
  7.             capacity -= w[i]
    & H  O7 r7 H6 J5 Q6 ^1 d
  8.     x[0] = 1 if (m[1][capacity] > 0) else 0
复制代码
在这一部分,代码反向遍历动态规划表,从最后一行向前找到解的路径。如果 m[capacity] 等于 m[i-1][capacity],表示第 i 个物品没有放入背包,否则放入背包,并更新剩余容量 capacity。
  1.     weight = 04 ]7 G& n+ ^# q\" L- y3 ]! Q. F
  2.     value = 0: h# E3 ^; H3 f7 M+ a\" Z1 M
  3.     print('装载的物品编号为:')
    : t5 }5 r1 M3 F4 g
  4.     for i in range(len(x)):5 g8 W$ Q0 B( u3 x4 ]3 L
  5.         if (x[i] == 1):
    ; {( Q# T* c: f5 i! [
  6.             weight = weight + w[i]
    : y! G, W4 D% a( ^1 p7 y\" G
  7.             value = value + v[i]3 \5 _; T# e) C3 Z3 O* l: Q
  8.             print(' ', i + 1)
    9 ]# `/ Y\" v7 @1 A! g6 w
  9.     print('装载的物品重量为:')
    9 M; o1 d) Y$ F. L* w
  10.     print(weight)
    6 F7 g! T9 t; O1 K+ C
  11.     print('装入的物品价值为:'): j3 K0 U$ B7 {5 @* \
  12.     print(value)\" F6 k0 f4 q$ [) X' m
  13.     return m
复制代码
最后,代码计算了被选择的物品的总重量和总价值,并将它们打印出来。函数返回动态规划表 m。
' M( Z0 m* Z$ y& b/ p3 ~/ z这段代码实现了0-1背包问题的解决方法,它通过动态规划算法找到最优解,即在给定背包容量下可以获得的最大总价值,以及选择哪些物品放入背包。' y$ M/ k9 r9 ~

6 z5 l. N" X& m) s, X; n5 G+ R最后在函数的输入文件中,如下例,第一行物品数量为:5 背包载重量为:10物品的重量列表为:[2, 2, 6, 5, 4] 物品的价值列表为: [6, 3, 5, 4, 6]
. z1 ^/ j3 l# K  C: b
VeryCapture_20231107110045.jpg
+ y4 F" o' Q7 ^( c# B& [0 P1 O% z' L

5 Q6 }. h, E0 K& f& t4 J) h; N) J
接下来展示我们的输出结果:
7 e9 _" L2 R1 r8 C
VeryCapture_20231107110508.jpg
1 f; X7 Z6 d' u% f
, r2 Y( }, M! ]5 U

5 c+ a; q% S4 j1 F1 |) W8 I
具体代码如下:

$ {  G- N5 i+ U/ Y5 Y" F
+ o$ L8 t, n4 }: U* h) b
" P' Q8 ~6 [- p/ M8 ^
/ ?, U0 ?/ x* x5 i2 C
# g6 F: ~  Y5 h

; @# g: {) d+ D( e* [

基础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-10 15:24 , Processed in 0.820990 second(s), 55 queries .

回顶部