QQ登录

只需要一步,快速开始

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

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

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

1189

主题

4

听众

2934

积分

该用户从未签到

跳转到指定楼层
1#
发表于 2023-11-7 11:17 |只看该作者 |倒序浏览
|招呼Ta 关注Ta
一、首先介绍一下0-1背包问题:
; r  o7 I/ Z/ v) _- s2 j
        0-1背包问题(0-1 Knapsack Problem)是一个经典的组合优化问题,通常在计算机科学和运筹学中讨论。这个问题涉及到一个背包和一组物品,每个物品都有一个特定的重量和价值。问题的目标是在给定背包的最大容量下,选择一组物品放入背包,以使得所选物品的总重量不超过背包的容量,同时最大化这些物品的总价值。
- M2 t& I$ a& l, |( w& A         0-1背包问题的名称中的“0-1”表示每个物品要么完全放入背包(选择)要么完全不放入背包(不选择),不能部分放入。这是问题的一个关键特征,与分数背包问题不同,分数背包问题允许部分放入物品。4 O& p; K1 Y' v, Y7 R. v' d0 \
问题的形式描述如下:

+ Y6 c- P: S7 L( ]! z                 1.给定一个固定容量的背包,通常表示为一个正整数W(背包的最大承载重量)。- c" J) I7 O$ p. W% g7 b, e
                 2.给定一组物品,每个物品都有两个属性:重量(weight)和价值(value)。' f6 W0 _% V8 U  U+ E9 y. ~% D4 {
                 3.对每个物品,你可以选择将其放入背包(选择)或不放入背包(不选择)。% H7 t7 ^4 b5 m$ h% j
                 4.每个物品只能选择一次,即要么放入背包,要么不放入。
/ w& t; x4 H3 Q6 M" {+ c: [                 5.目标是选择一个物品组合,使得它们的总重量不超过背包容量W,同时使它们的总价值最大化。
2 |" f0 u* G8 }3 x- D8 p
4 R- L) w, Z* `3 _' c解决0-1背包问题的一种常见方法是使用动态规划(Dynamic Programming)算法。这个问题有广泛的应用,包括资源分配、排程问题、投资组合优化等领域。它还是计算复杂性理论中的一个经典问题,通常被用来说明NP难问题的概念。4 N" ~# d, i/ \2 P1 D8 m, |# l

1 n( r* w4 n, B1 a; Y1 a) y1 w4 Q$ B. U
二、 介绍代码- P, M" f/ g( I' E- J3 a5 t
这段代码是一个Python实现的0-1背包问题的解决方法,使用了动态规划算法来找到最优解。以下是对代码的详细解释:
  1. def knapsack(v, w, n, capacity):\" ?, `6 y! y# P; ~# V3 l! O
  2.     i = 0( L* ~9 |\" H; R0 {5 ~
  3.     capacity = capacity + 1  # 初始化背包容量最大值; f+ r6 |/ H3 e& d( m% @( X$ v
  4.     m = np.zeros((n, capacity))  # 初始化
    : m* E3 i1 y9 [
  5.     x = np.zeros(n)
复制代码
1.v 是一个长度为 n 的列表,表示每个物品的价值。
9 I" F! w2 ~& x( J9 W! ?) Y5 r2.w 是一个长度为 n 的列表,表示每个物品的重量。0 ^3 V: R3 B9 V! |( n) w
3.n 表示物品的数量。1 k; M8 ?, d4 a6 J$ M  d  G% F
4.capacity 表示背包的容量。
; T2 ^! l( [3 l* J/ ^" I8 Z0 z( g. |9 J/ Q4 Y; Q& R
代码首先初始化了一个二维数组 m 作为动态规划表,其中 m[j] 表示在考虑前 i 个物品时,背包容量为 j 时可以获得的最大总价值。数组 x 用来存储最终的解,x 表示是否选择第 i 个物品。
  1.     for i in range(n):
    # F' d: d) M+ ^- Z
  2.         for j in range(capacity):9 T\" t/ i# O9 e% s* G
  3.             if (j >= w[i]):
    . ^5 G\" ?  h% B
  4.                 m[i][j] = max(m[i - 1][j], m[i - 1][j - w[i]] + v[i])0 ?& [# C\" d: f
  5.             else:# m2 G, X0 @! A1 C9 C; B; g
  6.                 m[i][j] = m[i - 1][j]
复制代码
在这个部分,代码使用了一个嵌套的循环,遍历了所有物品和不同的背包容量。对于每个物品 i 和容量 j,代码计算了两种情况下的最大总价值:
) B+ Y5 T; d( J# ]( D& O5.如果当前物品的重量 w 小于等于当前容量 j,那么可以选择将第 i 个物品放入背包,此时总价值为 m[i-1][j-w] + v,或者选择不放入,此时总价值为 m[i-1][j]。代码选择其中较大的值作为 m[j]。4 T3 b/ a6 D0 ]) F7 S2 \+ t
6.如果当前物品的重量 w 大于当前容量 j,则无法放入物品,所以总价值等于上一行的值 m[i-1][j]。  Q0 o8 h7 [6 v6 h0 a( ]7 ^
  \' @* Z9 ?2 \2 }  t5 `, F
这个循环填充了动态规划表 m,最终 m[n-1][capacity] 包含了问题的最优解,即在给定容量下可以获得的最大总价值。
  1.     capacity = capacity - 19 c4 O' ~( G# K7 ~0 N3 e& P5 [- a
  2.     for i in range(n - 1, 0, -1):. A\" N6 Y4 T+ |% F5 Z
  3.         if (m[i][capacity] == m[i - 1][capacity]):; }1 _: e\" H% _* [2 Z
  4.             x[i] = 0
    : r7 |% L- K  S9 V; O( e
  5.         else:
    1 }3 W! ^9 Y: c* w; h1 A$ X
  6.             x[i] = 1. X7 ]! O; C. g  Y6 L+ ~, R1 v
  7.             capacity -= w[i]7 N3 h; _- E! Z% F/ F% C  E8 [
  8.     x[0] = 1 if (m[1][capacity] > 0) else 0
复制代码
在这一部分,代码反向遍历动态规划表,从最后一行向前找到解的路径。如果 m[capacity] 等于 m[i-1][capacity],表示第 i 个物品没有放入背包,否则放入背包,并更新剩余容量 capacity。
  1.     weight = 0
    1 T3 p) J3 {1 h  r
  2.     value = 0% s- R3 q\" \- I5 D  k7 u( t# J
  3.     print('装载的物品编号为:')6 L* P2 r. \6 n* ]2 I2 }& X, ^( o
  4.     for i in range(len(x)):4 c7 X; e9 J5 o# z& b; b
  5.         if (x[i] == 1):
    . C1 p$ t. y& g8 x6 d6 g( l# [. d
  6.             weight = weight + w[i]\" S& y: I' N\" g1 g) W
  7.             value = value + v[i]
    0 U/ F6 C6 c2 ]9 i6 o
  8.             print(' ', i + 1)9 ~4 \( O. V6 ?2 H9 Y8 j5 n
  9.     print('装载的物品重量为:')
    , g- a6 Z; ?; h9 {! u% Y
  10.     print(weight); E' A* ]8 u$ [/ t
  11.     print('装入的物品价值为:')
    - d+ j0 W, x7 _% ~
  12.     print(value)
    # Q8 c6 @! [0 W4 E: Z: b
  13.     return m
复制代码
最后,代码计算了被选择的物品的总重量和总价值,并将它们打印出来。函数返回动态规划表 m。6 y, g, b$ B; l! `0 O
这段代码实现了0-1背包问题的解决方法,它通过动态规划算法找到最优解,即在给定背包容量下可以获得的最大总价值,以及选择哪些物品放入背包。5 B. s. h6 Y. N" h7 a5 K

# \- h# |! H6 _! W8 j; S8 _/ z最后在函数的输入文件中,如下例,第一行物品数量为:5 背包载重量为:10物品的重量列表为:[2, 2, 6, 5, 4] 物品的价值列表为: [6, 3, 5, 4, 6]
$ ^! k1 a4 Q7 z/ w- F
VeryCapture_20231107110045.jpg
! R" l/ B0 H/ f  o' L; q

) ?9 ?; ?- ^$ |6 `! T6 X
接下来展示我们的输出结果:
, a# X5 o) L( \6 |1 `
VeryCapture_20231107110508.jpg
7 T! v6 V( c  m4 o( ]
3 _( ^0 e* {$ i
. k, J9 _) V/ w! s7 I* ]; O$ a; h
具体代码如下:

6 Y5 D7 O2 C3 _& B4 u+ e( t  s* B' z" b+ j; G0 w" G3 c1 V

, X- ~3 ]& |9 p! Y7 m+ {3 g6 H3 u
# E, y& _1 t4 p) T. n7 r: }
/ q' C7 M1 @6 F$ R; J( l

基础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 13:15 , Processed in 2.892436 second(s), 54 queries .

回顶部