QQ登录

只需要一步,快速开始

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

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

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

1186

主题

4

听众

2923

积分

该用户从未签到

跳转到指定楼层
1#
发表于 2023-11-7 11:17 |只看该作者 |倒序浏览
|招呼Ta 关注Ta
一、首先介绍一下0-1背包问题:
: _' \, R* E; D& P; k& {
        0-1背包问题(0-1 Knapsack Problem)是一个经典的组合优化问题,通常在计算机科学和运筹学中讨论。这个问题涉及到一个背包和一组物品,每个物品都有一个特定的重量和价值。问题的目标是在给定背包的最大容量下,选择一组物品放入背包,以使得所选物品的总重量不超过背包的容量,同时最大化这些物品的总价值。
' `1 m, m+ J4 b1 j3 p         0-1背包问题的名称中的“0-1”表示每个物品要么完全放入背包(选择)要么完全不放入背包(不选择),不能部分放入。这是问题的一个关键特征,与分数背包问题不同,分数背包问题允许部分放入物品。! k( L, k" _- b' o  o( w
问题的形式描述如下:
7 ?( L5 }$ |6 s! I* I% N
                 1.给定一个固定容量的背包,通常表示为一个正整数W(背包的最大承载重量)。
1 t8 N$ k! R; a: z9 m                 2.给定一组物品,每个物品都有两个属性:重量(weight)和价值(value)。
6 u( h$ i. _' I* w& o6 m9 o                 3.对每个物品,你可以选择将其放入背包(选择)或不放入背包(不选择)。2 P7 n7 x- l9 N. ?
                 4.每个物品只能选择一次,即要么放入背包,要么不放入。  Q( C* [4 d' w; w
                 5.目标是选择一个物品组合,使得它们的总重量不超过背包容量W,同时使它们的总价值最大化。
& a9 q; e$ b6 Y! Z, R' d. u( Y
5 h7 c  d$ I9 o解决0-1背包问题的一种常见方法是使用动态规划(Dynamic Programming)算法。这个问题有广泛的应用,包括资源分配、排程问题、投资组合优化等领域。它还是计算复杂性理论中的一个经典问题,通常被用来说明NP难问题的概念。
) W0 {- Q6 @' ]3 q3 E2 v1 o7 F! z" j, l9 Y

% r% f1 T/ y0 v) X$ D* y二、 介绍代码
+ h) w8 w5 H' u: t# }6 e这段代码是一个Python实现的0-1背包问题的解决方法,使用了动态规划算法来找到最优解。以下是对代码的详细解释:
  1. def knapsack(v, w, n, capacity):6 [5 F$ ~1 k' r' R5 A/ d$ r
  2.     i = 0\" I: E' N6 r3 p) n
  3.     capacity = capacity + 1  # 初始化背包容量最大值
    % J0 m4 ?# s1 r' {0 W; B/ Y
  4.     m = np.zeros((n, capacity))  # 初始化% J- Z0 _' P; b\" K- c
  5.     x = np.zeros(n)
复制代码
1.v 是一个长度为 n 的列表,表示每个物品的价值。
7 ]+ n- B' d! f0 r2.w 是一个长度为 n 的列表,表示每个物品的重量。
/ j# y5 q8 t6 s5 T/ y' \3.n 表示物品的数量。" @  x2 s( L' ~7 n8 p* `# ]2 t
4.capacity 表示背包的容量。
: R1 e0 Z. N$ E  c& J
5 N  V4 ]- n7 U, ]- l代码首先初始化了一个二维数组 m 作为动态规划表,其中 m[j] 表示在考虑前 i 个物品时,背包容量为 j 时可以获得的最大总价值。数组 x 用来存储最终的解,x 表示是否选择第 i 个物品。
  1.     for i in range(n):! W) k\" Y2 m\" k) a( Y
  2.         for j in range(capacity):
    2 `  c6 Z7 h  k) y. o7 e1 }
  3.             if (j >= w[i]):2 B8 N; V' l  K; y. H
  4.                 m[i][j] = max(m[i - 1][j], m[i - 1][j - w[i]] + v[i])
    0 E# q6 z\" P& i6 g
  5.             else:
    4 L5 a' T9 _\" J\" U
  6.                 m[i][j] = m[i - 1][j]
复制代码
在这个部分,代码使用了一个嵌套的循环,遍历了所有物品和不同的背包容量。对于每个物品 i 和容量 j,代码计算了两种情况下的最大总价值:
: m1 c( u* ^' V; \' p" ^4 W# H5.如果当前物品的重量 w 小于等于当前容量 j,那么可以选择将第 i 个物品放入背包,此时总价值为 m[i-1][j-w] + v,或者选择不放入,此时总价值为 m[i-1][j]。代码选择其中较大的值作为 m[j]。/ v- p! |/ o/ i9 @( N
6.如果当前物品的重量 w 大于当前容量 j,则无法放入物品,所以总价值等于上一行的值 m[i-1][j]。
3 u* c1 W$ W/ a% U
: t; D( d  N6 c' a这个循环填充了动态规划表 m,最终 m[n-1][capacity] 包含了问题的最优解,即在给定容量下可以获得的最大总价值。
  1.     capacity = capacity - 1
    ' X7 u; }- c* q* w# D6 L6 j
  2.     for i in range(n - 1, 0, -1):! E3 n2 h8 `+ S0 _
  3.         if (m[i][capacity] == m[i - 1][capacity]):% R4 A. r' _% t6 B, W0 \
  4.             x[i] = 0: R! v( R0 y- K' g- r. ?
  5.         else:: d( w6 `& `1 y& S- v/ u- D
  6.             x[i] = 1, u; S% G. J; S6 [5 ]& L
  7.             capacity -= w[i]0 |8 M/ i1 a) t2 p0 H1 U. T+ X\" _
  8.     x[0] = 1 if (m[1][capacity] > 0) else 0
复制代码
在这一部分,代码反向遍历动态规划表,从最后一行向前找到解的路径。如果 m[capacity] 等于 m[i-1][capacity],表示第 i 个物品没有放入背包,否则放入背包,并更新剩余容量 capacity。
  1.     weight = 0
    \" F* c: ^. Q* y& B2 I+ k* ]
  2.     value = 0
    % E5 ]. u4 e  b8 @
  3.     print('装载的物品编号为:')
    & U$ {' Y$ u8 o3 ]8 J* u' F$ k
  4.     for i in range(len(x)):
    & }7 K3 I: J- N0 k/ t* H
  5.         if (x[i] == 1):1 K4 F9 x\" |- k6 i' m3 ?
  6.             weight = weight + w[i]
    ( z/ g\" q; u% ?- G2 ^
  7.             value = value + v[i]
    ) o( d; ]. v  e/ R9 I! l, C0 `
  8.             print(' ', i + 1)
    ) B0 Y\" D' `# x+ \: ^6 a( y
  9.     print('装载的物品重量为:')  c- ~  G4 T: |+ J: k3 H: d; h: j
  10.     print(weight)/ q5 \5 C. G, G7 T- u
  11.     print('装入的物品价值为:')
    % C& @0 O# M3 l8 @: p
  12.     print(value)
    * }. j2 s$ O& `: n1 z
  13.     return m
复制代码
最后,代码计算了被选择的物品的总重量和总价值,并将它们打印出来。函数返回动态规划表 m。
/ [$ Q0 m/ s. H4 N0 _# O( ?这段代码实现了0-1背包问题的解决方法,它通过动态规划算法找到最优解,即在给定背包容量下可以获得的最大总价值,以及选择哪些物品放入背包。
. j$ ]+ z$ p+ a% v
# a9 U# A6 v: ~: b. q; _最后在函数的输入文件中,如下例,第一行物品数量为:5 背包载重量为:10物品的重量列表为:[2, 2, 6, 5, 4] 物品的价值列表为: [6, 3, 5, 4, 6]" d# X4 v5 k4 q5 o/ F$ U9 n
VeryCapture_20231107110045.jpg
: u' X8 k/ J$ v  A" B
" D% L! b7 V5 q2 h0 U' g1 k
接下来展示我们的输出结果:

. x2 d4 D! K' G: J
VeryCapture_20231107110508.jpg

8 I) m6 ^; T$ k+ p, @# g4 o* _
" W/ Y. `& F: s- ?3 n0 i

+ Z* I, K4 y' `. j  e& N
具体代码如下:

3 R- @4 i+ L) Q1 \" b% Z$ O: J- K3 W- E# |2 W

6 n& J7 k8 p- u* Q' P0 M! Y$ O* V# Z! ~
; B3 z# a/ z! d6 k
/ I  |+ X) T7 ^- |5 s, ?
+ ~5 h6 Q# Y' X( v

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

回顶部