QQ登录

只需要一步,快速开始

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

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

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

1186

主题

4

听众

2922

积分

该用户从未签到

跳转到指定楼层
1#
发表于 2023-11-7 11:17 |只看该作者 |倒序浏览
|招呼Ta 关注Ta
一、首先介绍一下0-1背包问题:' Y6 E/ z( z0 _/ v+ V9 D8 g% x: Y
        0-1背包问题(0-1 Knapsack Problem)是一个经典的组合优化问题,通常在计算机科学和运筹学中讨论。这个问题涉及到一个背包和一组物品,每个物品都有一个特定的重量和价值。问题的目标是在给定背包的最大容量下,选择一组物品放入背包,以使得所选物品的总重量不超过背包的容量,同时最大化这些物品的总价值。
/ |9 h' k9 {: Z         0-1背包问题的名称中的“0-1”表示每个物品要么完全放入背包(选择)要么完全不放入背包(不选择),不能部分放入。这是问题的一个关键特征,与分数背包问题不同,分数背包问题允许部分放入物品。
0 T7 |% m* C, d! q- v
问题的形式描述如下:

% ]5 A8 p1 @* O- r1 o$ ]1 i9 n                 1.给定一个固定容量的背包,通常表示为一个正整数W(背包的最大承载重量)。
$ \( ?2 s2 v8 V8 V4 r                 2.给定一组物品,每个物品都有两个属性:重量(weight)和价值(value)。
: p) N# A' i7 g* R& i                 3.对每个物品,你可以选择将其放入背包(选择)或不放入背包(不选择)。% B/ l5 F, A. ?* |( \
                 4.每个物品只能选择一次,即要么放入背包,要么不放入。9 {; n8 v6 N3 G/ c4 A
                 5.目标是选择一个物品组合,使得它们的总重量不超过背包容量W,同时使它们的总价值最大化。
  Y. F  C" m. F; m
! `, Q6 s2 c0 D3 @解决0-1背包问题的一种常见方法是使用动态规划(Dynamic Programming)算法。这个问题有广泛的应用,包括资源分配、排程问题、投资组合优化等领域。它还是计算复杂性理论中的一个经典问题,通常被用来说明NP难问题的概念。
1 g& x6 j! L* _0 s$ x7 e. h3 I8 s  D1 k6 |# J0 w

4 o: b1 v, I9 d二、 介绍代码
3 R+ n8 C, h. T. O5 D9 E. C这段代码是一个Python实现的0-1背包问题的解决方法,使用了动态规划算法来找到最优解。以下是对代码的详细解释:
  1. def knapsack(v, w, n, capacity):0 O\" D; X, `9 q$ `4 h% N
  2.     i = 0\" r& _  t; r% h8 w% y& V
  3.     capacity = capacity + 1  # 初始化背包容量最大值. `2 }) K. B& z* ^
  4.     m = np.zeros((n, capacity))  # 初始化
    ( I; M6 Y+ w: K' W\" Y5 w% E+ M
  5.     x = np.zeros(n)
复制代码
1.v 是一个长度为 n 的列表,表示每个物品的价值。
1 z5 c% B4 \* t5 U1 z& C( S2.w 是一个长度为 n 的列表,表示每个物品的重量。, h, _8 M8 z9 g
3.n 表示物品的数量。4 H  Q1 E( H7 ~3 V7 ]. |
4.capacity 表示背包的容量。- q( P5 {$ t0 X: }7 ~& S
$ N; ^' W4 ]5 A
代码首先初始化了一个二维数组 m 作为动态规划表,其中 m[j] 表示在考虑前 i 个物品时,背包容量为 j 时可以获得的最大总价值。数组 x 用来存储最终的解,x 表示是否选择第 i 个物品。
  1.     for i in range(n):7 G! L\" o/ t3 J1 G9 T
  2.         for j in range(capacity):
    $ ~% W# I) e% r\" y  x, n) v4 G
  3.             if (j >= w[i]):
    ; t* ]1 m- l3 i  B: z7 [* E# C
  4.                 m[i][j] = max(m[i - 1][j], m[i - 1][j - w[i]] + v[i])+ U$ i( F* `: E! T, E1 }
  5.             else:+ e' ~; e: H- B( l
  6.                 m[i][j] = m[i - 1][j]
复制代码
在这个部分,代码使用了一个嵌套的循环,遍历了所有物品和不同的背包容量。对于每个物品 i 和容量 j,代码计算了两种情况下的最大总价值:
: x9 _% f' W- M# ^2 i5.如果当前物品的重量 w 小于等于当前容量 j,那么可以选择将第 i 个物品放入背包,此时总价值为 m[i-1][j-w] + v,或者选择不放入,此时总价值为 m[i-1][j]。代码选择其中较大的值作为 m[j]。; t9 ]" ]; v  D! r
6.如果当前物品的重量 w 大于当前容量 j,则无法放入物品,所以总价值等于上一行的值 m[i-1][j]。
% c& [0 J9 O! I  ]" f+ l. r& D
这个循环填充了动态规划表 m,最终 m[n-1][capacity] 包含了问题的最优解,即在给定容量下可以获得的最大总价值。
  1.     capacity = capacity - 1  \* Z6 N0 d, i* d& b( ]& D6 k
  2.     for i in range(n - 1, 0, -1):
    \" s6 E( x/ `$ N. V0 n
  3.         if (m[i][capacity] == m[i - 1][capacity]):
      d7 i$ N; O1 u: G! P7 C
  4.             x[i] = 0* A1 N, t# E8 {0 T# P) {# r
  5.         else:: U* Z5 K  u3 g# l( {7 {6 a
  6.             x[i] = 1
    ! _, U& C8 _8 X\" ^9 |/ ^; m7 |
  7.             capacity -= w[i]
    ; M8 N4 }) i- y! f6 @4 m* x
  8.     x[0] = 1 if (m[1][capacity] > 0) else 0
复制代码
在这一部分,代码反向遍历动态规划表,从最后一行向前找到解的路径。如果 m[capacity] 等于 m[i-1][capacity],表示第 i 个物品没有放入背包,否则放入背包,并更新剩余容量 capacity。
  1.     weight = 0) l3 [+ n- T6 r' c. O- [
  2.     value = 0; \1 y: n+ N( u1 S. g% b
  3.     print('装载的物品编号为:')
    & t$ ]8 |: j8 |5 z
  4.     for i in range(len(x)):
    % A  {7 I, O; l& E8 k* n- s4 U! G\" i
  5.         if (x[i] == 1):. f/ o& @$ ~. d! D6 [$ E
  6.             weight = weight + w[i]* N  c# q* w+ }# D
  7.             value = value + v[i]
    # q( i6 p' }# [. N' p5 B
  8.             print(' ', i + 1)$ h& m\" Y2 Q; x
  9.     print('装载的物品重量为:')/ l( x0 n7 e# d: [
  10.     print(weight)7 e  |9 Y& O\" h# f) A: I9 e5 f
  11.     print('装入的物品价值为:')
    7 f+ q2 X% B6 j4 W. u* E
  12.     print(value)
    + L, Y9 L/ J/ N& F2 i
  13.     return m
复制代码
最后,代码计算了被选择的物品的总重量和总价值,并将它们打印出来。函数返回动态规划表 m。' z$ a1 ^9 D; B3 R- T0 [, y
这段代码实现了0-1背包问题的解决方法,它通过动态规划算法找到最优解,即在给定背包容量下可以获得的最大总价值,以及选择哪些物品放入背包。$ `3 N8 Z$ n3 l, d8 i
( T! u4 R- y: D7 I
最后在函数的输入文件中,如下例,第一行物品数量为:5 背包载重量为:10物品的重量列表为:[2, 2, 6, 5, 4] 物品的价值列表为: [6, 3, 5, 4, 6]
2 J$ I6 v# A# T; w7 ?" p( d* h1 \5 m+ o
VeryCapture_20231107110045.jpg

/ c. j% {7 g& T5 g
  _, F2 B" [# r+ G" J* v: H
接下来展示我们的输出结果:
6 r( w3 ^. t: {" _3 E- J4 P2 q
VeryCapture_20231107110508.jpg
# X, T4 U+ @6 Y6 }
6 @0 i3 n. R4 X

" w7 F& a& d( f
具体代码如下:
1 ~. j8 |4 x, J; `. F* u6 j

; Q+ I4 q' K) x2 ^3 x
9 L# D) V4 _% C% ]& n$ A$ @" n; f6 g" D- z+ X9 g9 i3 r  m7 R
7 W- T$ s5 I- Y
0 u2 b/ n3 |5 e8 Z- A: ?3 a

基础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 16:59 , Processed in 0.327592 second(s), 55 queries .

回顶部