QQ登录

只需要一步,快速开始

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

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

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

1189

主题

4

听众

2934

积分

该用户从未签到

跳转到指定楼层
1#
发表于 2023-11-7 11:17 |只看该作者 |倒序浏览
|招呼Ta 关注Ta
一、首先介绍一下0-1背包问题:7 s* O0 a6 Z. ~+ I
        0-1背包问题(0-1 Knapsack Problem)是一个经典的组合优化问题,通常在计算机科学和运筹学中讨论。这个问题涉及到一个背包和一组物品,每个物品都有一个特定的重量和价值。问题的目标是在给定背包的最大容量下,选择一组物品放入背包,以使得所选物品的总重量不超过背包的容量,同时最大化这些物品的总价值。
' V2 f- f9 [* O/ I) ?         0-1背包问题的名称中的“0-1”表示每个物品要么完全放入背包(选择)要么完全不放入背包(不选择),不能部分放入。这是问题的一个关键特征,与分数背包问题不同,分数背包问题允许部分放入物品。
, f' N' {2 D, x% k" k! |  M% E- g6 J
问题的形式描述如下:

3 W6 ]4 `; T$ v                 1.给定一个固定容量的背包,通常表示为一个正整数W(背包的最大承载重量)。
0 O9 a' X9 T& m                 2.给定一组物品,每个物品都有两个属性:重量(weight)和价值(value)。
; g% f9 G% i: x. [* y* M* K                 3.对每个物品,你可以选择将其放入背包(选择)或不放入背包(不选择)。
7 G% G; f+ l% v( e                 4.每个物品只能选择一次,即要么放入背包,要么不放入。
8 |2 w1 ~; |* ]; P: ^                 5.目标是选择一个物品组合,使得它们的总重量不超过背包容量W,同时使它们的总价值最大化。
, i1 z$ M0 O4 s& n. v
5 Q/ h5 ^  c- A9 r! P解决0-1背包问题的一种常见方法是使用动态规划(Dynamic Programming)算法。这个问题有广泛的应用,包括资源分配、排程问题、投资组合优化等领域。它还是计算复杂性理论中的一个经典问题,通常被用来说明NP难问题的概念。
+ |* m3 c( v1 i/ x0 ^" x& ^' `; {3 B# f, v

- H% @) O4 D7 b  E二、 介绍代码$ K# {  n: Q5 q
这段代码是一个Python实现的0-1背包问题的解决方法,使用了动态规划算法来找到最优解。以下是对代码的详细解释:
  1. def knapsack(v, w, n, capacity):
    & m/ Q; a; S& P9 i5 Q: @7 k
  2.     i = 0
    * d* ^: @5 {1 u/ U$ `+ c
  3.     capacity = capacity + 1  # 初始化背包容量最大值( M/ D, j- v, v4 V
  4.     m = np.zeros((n, capacity))  # 初始化7 I. Y3 P% f2 `8 r) d) M0 c2 @9 L
  5.     x = np.zeros(n)
复制代码
1.v 是一个长度为 n 的列表,表示每个物品的价值。/ b# [" F, B( W& Y/ _! ~: M3 _
2.w 是一个长度为 n 的列表,表示每个物品的重量。; R5 V8 w: `' u
3.n 表示物品的数量。
* F8 f" ]+ b/ p& l' Q' R1 Z# O0 W2 X' x4.capacity 表示背包的容量。
( J& m" ?/ d. _( \* K( p5 y+ V- g1 _7 {$ Y6 P/ J
代码首先初始化了一个二维数组 m 作为动态规划表,其中 m[j] 表示在考虑前 i 个物品时,背包容量为 j 时可以获得的最大总价值。数组 x 用来存储最终的解,x 表示是否选择第 i 个物品。
  1.     for i in range(n):
    3 M# p) u/ Y( n. c- t: ^
  2.         for j in range(capacity):7 N6 X2 Y  u9 ]% F. ?; h
  3.             if (j >= w[i]):
    , a6 {; H& F+ w& C! V$ U
  4.                 m[i][j] = max(m[i - 1][j], m[i - 1][j - w[i]] + v[i])
    : V( M2 e  l) N! F& ^4 N
  5.             else:  {5 u, r6 p3 [( u4 O  {
  6.                 m[i][j] = m[i - 1][j]
复制代码
在这个部分,代码使用了一个嵌套的循环,遍历了所有物品和不同的背包容量。对于每个物品 i 和容量 j,代码计算了两种情况下的最大总价值:+ g/ w3 h$ N' B+ T3 r
5.如果当前物品的重量 w 小于等于当前容量 j,那么可以选择将第 i 个物品放入背包,此时总价值为 m[i-1][j-w] + v,或者选择不放入,此时总价值为 m[i-1][j]。代码选择其中较大的值作为 m[j]。* y7 u; K/ `1 Z2 w& e
6.如果当前物品的重量 w 大于当前容量 j,则无法放入物品,所以总价值等于上一行的值 m[i-1][j]。
6 `8 q" y% s. }' s
8 G1 K3 \9 X' I这个循环填充了动态规划表 m,最终 m[n-1][capacity] 包含了问题的最优解,即在给定容量下可以获得的最大总价值。
  1.     capacity = capacity - 1) z6 g8 i8 {8 q0 |
  2.     for i in range(n - 1, 0, -1):
    4 H# R1 j, k- i0 ^3 a# l# B
  3.         if (m[i][capacity] == m[i - 1][capacity]):- n3 z& c. H7 w; x7 Y* b0 c
  4.             x[i] = 0& o: L7 U8 ^3 r3 m2 Y4 Z5 h3 B
  5.         else:
    ; B7 d) V, L7 K
  6.             x[i] = 1, M2 p- ^; {; d4 D) A/ D) N. i
  7.             capacity -= w[i]
    4 M' ^; h4 _1 p+ f
  8.     x[0] = 1 if (m[1][capacity] > 0) else 0
复制代码
在这一部分,代码反向遍历动态规划表,从最后一行向前找到解的路径。如果 m[capacity] 等于 m[i-1][capacity],表示第 i 个物品没有放入背包,否则放入背包,并更新剩余容量 capacity。
  1.     weight = 0. ]\" ?2 B: g- w\" i
  2.     value = 0- U' L4 c% m2 T+ }
  3.     print('装载的物品编号为:')
    3 r$ ]% l  A6 C, R/ Y1 ?+ P6 x
  4.     for i in range(len(x)):
    7 e6 w( p; ]: i\" e( ^/ i) `
  5.         if (x[i] == 1):
    1 r8 y+ F+ \7 s, z8 a% _
  6.             weight = weight + w[i]; u: d, S* a8 v4 U5 Z
  7.             value = value + v[i]
    : x7 i, q6 |) \! m& C
  8.             print(' ', i + 1)( |/ n- o- Y, {
  9.     print('装载的物品重量为:')
    - I. H+ M- H/ \
  10.     print(weight)
      d7 A\" g! x( Q6 l+ p6 K
  11.     print('装入的物品价值为:')! g2 {& E4 C, K+ A
  12.     print(value)- B3 i% y3 {2 i! H) R\" c
  13.     return m
复制代码
最后,代码计算了被选择的物品的总重量和总价值,并将它们打印出来。函数返回动态规划表 m。5 P) N; j' I% d6 }
这段代码实现了0-1背包问题的解决方法,它通过动态规划算法找到最优解,即在给定背包容量下可以获得的最大总价值,以及选择哪些物品放入背包。; [9 H$ U. E4 a3 l; M7 ^1 [

4 ?0 O/ A: O4 d6 X# C最后在函数的输入文件中,如下例,第一行物品数量为:5 背包载重量为:10物品的重量列表为:[2, 2, 6, 5, 4] 物品的价值列表为: [6, 3, 5, 4, 6]4 F$ q3 t/ X: j) T( v7 [. c
VeryCapture_20231107110045.jpg

+ j3 P% c" L" r' N! P9 L* n: i* S3 H1 A. S) U7 e+ j
接下来展示我们的输出结果:

3 Q; J3 k& R7 F1 V% E* V
VeryCapture_20231107110508.jpg

' L# n  D% s: s8 C* u
& ~# D/ R. ?; O" k* [
8 ~3 _' q- v  g: E3 O
具体代码如下:

; W5 ?" l; L8 E7 D2 ~
, Y/ U# v. `( A4 {" O' r/ _9 x' F: U0 s! j6 Y" e
& o! J0 |( e) f8 h& H& o6 |
3 E* D$ V4 ^2 M& j5 u- `- d

* {5 u# P1 F" u4 s0 N9 Z

基础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 08:49 , Processed in 1.975285 second(s), 54 queries .

回顶部