QQ登录

只需要一步,快速开始

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

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

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

1186

主题

4

听众

2922

积分

该用户从未签到

跳转到指定楼层
1#
发表于 2023-11-7 11:17 |只看该作者 |倒序浏览
|招呼Ta 关注Ta
一、首先介绍一下0-1背包问题:+ b' q  W& i) K/ d- O- @4 L
        0-1背包问题(0-1 Knapsack Problem)是一个经典的组合优化问题,通常在计算机科学和运筹学中讨论。这个问题涉及到一个背包和一组物品,每个物品都有一个特定的重量和价值。问题的目标是在给定背包的最大容量下,选择一组物品放入背包,以使得所选物品的总重量不超过背包的容量,同时最大化这些物品的总价值。3 Z/ E/ ]5 ^2 y) H0 E
         0-1背包问题的名称中的“0-1”表示每个物品要么完全放入背包(选择)要么完全不放入背包(不选择),不能部分放入。这是问题的一个关键特征,与分数背包问题不同,分数背包问题允许部分放入物品。. D- ~+ v9 Z5 S7 Z2 c6 h
问题的形式描述如下:
: ?* \$ N( a; X7 ~
                 1.给定一个固定容量的背包,通常表示为一个正整数W(背包的最大承载重量)。
: {$ y- z9 g' L) B  K                 2.给定一组物品,每个物品都有两个属性:重量(weight)和价值(value)。
  j: a7 _4 h, v$ [                 3.对每个物品,你可以选择将其放入背包(选择)或不放入背包(不选择)。7 L3 R- l6 M: x; G
                 4.每个物品只能选择一次,即要么放入背包,要么不放入。
( m( _, w3 |8 z' @' s4 Z7 V                 5.目标是选择一个物品组合,使得它们的总重量不超过背包容量W,同时使它们的总价值最大化。6 a. I& E* W. u. I1 X" i
8 F: o; h- J, ?% ]5 u
解决0-1背包问题的一种常见方法是使用动态规划(Dynamic Programming)算法。这个问题有广泛的应用,包括资源分配、排程问题、投资组合优化等领域。它还是计算复杂性理论中的一个经典问题,通常被用来说明NP难问题的概念。0 r9 n+ [$ ~+ z6 |% }# J; i

4 s: ^6 }! i+ t' V8 L( s) {9 Y" x
/ a- i1 W) x. u+ N& G9 L, a9 g二、 介绍代码3 J0 t" o7 R% Z, E- `/ E2 I4 A
这段代码是一个Python实现的0-1背包问题的解决方法,使用了动态规划算法来找到最优解。以下是对代码的详细解释:
  1. def knapsack(v, w, n, capacity):* Z. U2 P6 u! I8 @7 V6 E
  2.     i = 0
    7 K: f6 d' y- {6 l$ d
  3.     capacity = capacity + 1  # 初始化背包容量最大值' J& C3 `7 D; Y1 `6 E
  4.     m = np.zeros((n, capacity))  # 初始化
    ) n4 i6 D- G- }9 N5 p
  5.     x = np.zeros(n)
复制代码
1.v 是一个长度为 n 的列表,表示每个物品的价值。
+ u) |% e9 G4 H( P: y& L: g1 y" T# P2.w 是一个长度为 n 的列表,表示每个物品的重量。2 G1 t4 G. C3 \, Q- ~
3.n 表示物品的数量。
3 L' O# d+ ]6 T/ c7 d4.capacity 表示背包的容量。
, ^4 m* f7 e, w' M2 m) w' b% B: }- f* W9 B' ?' a! Z' a) j" E4 \
代码首先初始化了一个二维数组 m 作为动态规划表,其中 m[j] 表示在考虑前 i 个物品时,背包容量为 j 时可以获得的最大总价值。数组 x 用来存储最终的解,x 表示是否选择第 i 个物品。
  1.     for i in range(n):) Q3 G, @, a* V$ c4 s8 k- T! |
  2.         for j in range(capacity):
      s. a2 E/ O+ x' u% h
  3.             if (j >= w[i]):6 }/ v1 d6 j- u% u4 `$ F
  4.                 m[i][j] = max(m[i - 1][j], m[i - 1][j - w[i]] + v[i])3 _3 T4 F/ I# Q$ f7 |1 j4 r
  5.             else:
    # ^( a7 j6 s+ T0 ?* F) o2 |2 w7 {
  6.                 m[i][j] = m[i - 1][j]
复制代码
在这个部分,代码使用了一个嵌套的循环,遍历了所有物品和不同的背包容量。对于每个物品 i 和容量 j,代码计算了两种情况下的最大总价值:
7 ?! d5 d. m4 [: d) e# P5.如果当前物品的重量 w 小于等于当前容量 j,那么可以选择将第 i 个物品放入背包,此时总价值为 m[i-1][j-w] + v,或者选择不放入,此时总价值为 m[i-1][j]。代码选择其中较大的值作为 m[j]。
/ P2 g% U! i! [0 j6.如果当前物品的重量 w 大于当前容量 j,则无法放入物品,所以总价值等于上一行的值 m[i-1][j]。
& @6 f+ v3 P0 `1 Z6 r% {* c8 w" g# K. \) o
这个循环填充了动态规划表 m,最终 m[n-1][capacity] 包含了问题的最优解,即在给定容量下可以获得的最大总价值。
  1.     capacity = capacity - 1
    \" [) R3 T4 e7 v# H. b, R
  2.     for i in range(n - 1, 0, -1):
    % h% h, m* c' i9 G# n% I9 r
  3.         if (m[i][capacity] == m[i - 1][capacity]):
    ( s& ^5 _+ y/ F2 e' X
  4.             x[i] = 0
    : q6 T- p8 s1 i
  5.         else:7 j; e0 m6 g. _. d8 q& \: s
  6.             x[i] = 15 L  a& M. {% c% H( B
  7.             capacity -= w[i]8 ~; ~; C\" D8 O
  8.     x[0] = 1 if (m[1][capacity] > 0) else 0
复制代码
在这一部分,代码反向遍历动态规划表,从最后一行向前找到解的路径。如果 m[capacity] 等于 m[i-1][capacity],表示第 i 个物品没有放入背包,否则放入背包,并更新剩余容量 capacity。
  1.     weight = 0
    2 D0 P, b* D0 L! Y\" E7 S
  2.     value = 02 P\" n/ s4 e( C4 \! c
  3.     print('装载的物品编号为:')
    + \( O0 F8 s0 T; J# P0 S3 [# I: \
  4.     for i in range(len(x)):. t6 ?5 P) C3 K2 h
  5.         if (x[i] == 1):# x' V! z, F# A+ w2 G7 s
  6.             weight = weight + w[i]
    , y5 W: o, g$ x: q( `8 S
  7.             value = value + v[i]
    \" i! w. J! A4 B9 L4 n, v9 d( q
  8.             print(' ', i + 1)
    1 |7 a, V\" @8 T6 A- L* ~% p/ i) P8 x1 H
  9.     print('装载的物品重量为:')
    4 V; P' g; p$ f6 ^8 Y
  10.     print(weight)
    % u) m9 U8 |6 R! x  j; n5 S  A
  11.     print('装入的物品价值为:')+ f8 p\" X4 J) t/ [! Y$ a& ]/ {
  12.     print(value)
    % q$ t; g. R! H: a- L
  13.     return m
复制代码
最后,代码计算了被选择的物品的总重量和总价值,并将它们打印出来。函数返回动态规划表 m。8 @& p2 ^4 R8 T* g$ v4 M
这段代码实现了0-1背包问题的解决方法,它通过动态规划算法找到最优解,即在给定背包容量下可以获得的最大总价值,以及选择哪些物品放入背包。
" T) i! I3 s: z2 }7 Q% q) M9 n$ l) V  [+ g4 Z- R. C9 Q7 k
最后在函数的输入文件中,如下例,第一行物品数量为:5 背包载重量为:10物品的重量列表为:[2, 2, 6, 5, 4] 物品的价值列表为: [6, 3, 5, 4, 6]1 s. K% g9 M1 M# a1 f+ _
VeryCapture_20231107110045.jpg
0 U* A$ L& ]' H* i# r* |

1 _. o" `% d5 r, F% L5 X
接下来展示我们的输出结果:

" n. w$ c6 ?; s9 R9 K  s
VeryCapture_20231107110508.jpg
4 d& v. Z: E; F! B) N$ k

. H+ p: n' p0 V0 g- h

, ^4 u7 {, _1 T( y6 K( Y
具体代码如下:
, a4 C( M: g4 a' J+ t6 C
) [5 o9 {/ v1 X/ j9 `. [5 [
. h/ l( d, ?% K4 _& y+ ~

5 \! j+ @9 @* g& a6 \2 N% b# b

% ?/ M: K' y3 p2 ?$ K
" E0 J! j2 G" V! m

基础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 13:09 , Processed in 0.403976 second(s), 59 queries .

回顶部