QQ登录

只需要一步,快速开始

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

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

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

1186

主题

4

听众

2923

积分

该用户从未签到

跳转到指定楼层
1#
发表于 2023-11-7 11:17 |只看该作者 |倒序浏览
|招呼Ta 关注Ta
一、首先介绍一下0-1背包问题:& v" X" e$ ~4 j6 Y/ [: d
        0-1背包问题(0-1 Knapsack Problem)是一个经典的组合优化问题,通常在计算机科学和运筹学中讨论。这个问题涉及到一个背包和一组物品,每个物品都有一个特定的重量和价值。问题的目标是在给定背包的最大容量下,选择一组物品放入背包,以使得所选物品的总重量不超过背包的容量,同时最大化这些物品的总价值。
6 q; U* r8 ]" y4 e+ L* P  |% g         0-1背包问题的名称中的“0-1”表示每个物品要么完全放入背包(选择)要么完全不放入背包(不选择),不能部分放入。这是问题的一个关键特征,与分数背包问题不同,分数背包问题允许部分放入物品。, c- C3 N" y  l: s: |: [+ v# Z
问题的形式描述如下:
) N" x/ U2 D0 k
                 1.给定一个固定容量的背包,通常表示为一个正整数W(背包的最大承载重量)。
! J& d+ d) J4 o; S8 k                 2.给定一组物品,每个物品都有两个属性:重量(weight)和价值(value)。
* h* a$ a( G6 ^2 G* N, _                 3.对每个物品,你可以选择将其放入背包(选择)或不放入背包(不选择)。, R: h) D/ L' q, e
                 4.每个物品只能选择一次,即要么放入背包,要么不放入。
  a. X8 D/ c( ^9 t6 }                 5.目标是选择一个物品组合,使得它们的总重量不超过背包容量W,同时使它们的总价值最大化。# }1 b- m* {0 p7 g$ P3 K

9 P  ?1 R5 R3 U9 a1 i9 p* I1 k' {解决0-1背包问题的一种常见方法是使用动态规划(Dynamic Programming)算法。这个问题有广泛的应用,包括资源分配、排程问题、投资组合优化等领域。它还是计算复杂性理论中的一个经典问题,通常被用来说明NP难问题的概念。' `' z* N' q! }( g+ f- S/ x
/ J% W7 z: x7 s6 ?4 W( S

: C; L5 o, `' `4 X+ b二、 介绍代码
4 ~! u% h) X/ V' c: i+ Z* \* W6 F这段代码是一个Python实现的0-1背包问题的解决方法,使用了动态规划算法来找到最优解。以下是对代码的详细解释:
  1. def knapsack(v, w, n, capacity):
    7 w; L6 y\" R$ S  l( `/ u
  2.     i = 0
    9 y0 V  p; D* b/ i5 T
  3.     capacity = capacity + 1  # 初始化背包容量最大值
    / c7 @+ v: v: [! Z& z& i! ]2 N
  4.     m = np.zeros((n, capacity))  # 初始化
    0 u0 F2 l3 m1 V2 F3 p\" m
  5.     x = np.zeros(n)
复制代码
1.v 是一个长度为 n 的列表,表示每个物品的价值。3 ]( D% R0 m# @# v. m; `" c0 X7 v
2.w 是一个长度为 n 的列表,表示每个物品的重量。
4 z& b1 A* N1 r1 M- l3.n 表示物品的数量。) L9 ?/ n1 z4 {/ E3 e0 R6 r8 r9 S9 l" t
4.capacity 表示背包的容量。
8 t8 T* ?5 F' C3 M
' S  x3 a7 t0 }代码首先初始化了一个二维数组 m 作为动态规划表,其中 m[j] 表示在考虑前 i 个物品时,背包容量为 j 时可以获得的最大总价值。数组 x 用来存储最终的解,x 表示是否选择第 i 个物品。
  1.     for i in range(n):( D$ \\" ]2 b: ]6 f
  2.         for j in range(capacity):
    - u% s\" U9 T- T$ d* N: f' ~9 L
  3.             if (j >= w[i]):
    3 T( M, F5 Y- s' H\" Z, @+ o8 P
  4.                 m[i][j] = max(m[i - 1][j], m[i - 1][j - w[i]] + v[i]). a/ U1 q1 s7 ~
  5.             else:$ q; o3 R0 k0 U9 ~) V\" {
  6.                 m[i][j] = m[i - 1][j]
复制代码
在这个部分,代码使用了一个嵌套的循环,遍历了所有物品和不同的背包容量。对于每个物品 i 和容量 j,代码计算了两种情况下的最大总价值:
: x& `( o1 c2 n5 ^5.如果当前物品的重量 w 小于等于当前容量 j,那么可以选择将第 i 个物品放入背包,此时总价值为 m[i-1][j-w] + v,或者选择不放入,此时总价值为 m[i-1][j]。代码选择其中较大的值作为 m[j]。
" Z( v& W2 H* R6.如果当前物品的重量 w 大于当前容量 j,则无法放入物品,所以总价值等于上一行的值 m[i-1][j]。% K2 {; }- M( a6 q% g) b' G  n" H
/ m1 @( H9 W, y
这个循环填充了动态规划表 m,最终 m[n-1][capacity] 包含了问题的最优解,即在给定容量下可以获得的最大总价值。
  1.     capacity = capacity - 1% |7 _6 Q  N# ^' w% P  k3 b, L
  2.     for i in range(n - 1, 0, -1):5 C6 c3 i: T& S4 G+ [
  3.         if (m[i][capacity] == m[i - 1][capacity]):
    3 C1 d. e6 J* k
  4.             x[i] = 07 v2 C6 o8 Q& x) Z& v, E8 A# f+ q) I
  5.         else:2 m% D! [7 k5 A7 E% S
  6.             x[i] = 1' l$ y! b* @: y1 p9 ^
  7.             capacity -= w[i]
    / Z. X0 A8 Z2 c- I. w! s$ D* S) n
  8.     x[0] = 1 if (m[1][capacity] > 0) else 0
复制代码
在这一部分,代码反向遍历动态规划表,从最后一行向前找到解的路径。如果 m[capacity] 等于 m[i-1][capacity],表示第 i 个物品没有放入背包,否则放入背包,并更新剩余容量 capacity。
  1.     weight = 0
    7 }/ F/ F* d. \- f' A
  2.     value = 0- `% G$ L+ n& r, e/ I
  3.     print('装载的物品编号为:')
    \" e$ q\" h0 t3 \7 O& l
  4.     for i in range(len(x)):- V3 ]% Q9 U9 u* T# D! W
  5.         if (x[i] == 1):% g5 x! n  D/ }
  6.             weight = weight + w[i]
    9 T' p& K+ v, ^* y
  7.             value = value + v[i]\" v' w: _4 x, z7 C( x+ |4 \
  8.             print(' ', i + 1)  d; w2 v. V! i! Z
  9.     print('装载的物品重量为:')
    ! @\" p% r/ K& Q9 H1 [, ~
  10.     print(weight)
    1 ^; R\" `4 Q* ]\" L) h
  11.     print('装入的物品价值为:')
    & E# v3 z$ Z% W
  12.     print(value)
    $ M. R- _1 P8 ?\" |. V. @2 D
  13.     return m
复制代码
最后,代码计算了被选择的物品的总重量和总价值,并将它们打印出来。函数返回动态规划表 m。; V. ^9 h# d9 k( |, p* k7 f: p* n
这段代码实现了0-1背包问题的解决方法,它通过动态规划算法找到最优解,即在给定背包容量下可以获得的最大总价值,以及选择哪些物品放入背包。
7 |0 Q, |8 e5 L+ P0 N7 Z
  `) x! O" h6 T8 B" y最后在函数的输入文件中,如下例,第一行物品数量为:5 背包载重量为:10物品的重量列表为:[2, 2, 6, 5, 4] 物品的价值列表为: [6, 3, 5, 4, 6]; |; @5 l" U) T) [5 u
VeryCapture_20231107110045.jpg

+ u5 n7 D+ Q9 z4 T1 j
; Q# M3 P2 Y/ G0 o& h% K" Z
接下来展示我们的输出结果:

7 H$ e7 u4 P- ]$ V  N
VeryCapture_20231107110508.jpg
; X$ x, v, Q/ f; f% Q
. g1 m, \2 U* B3 T
5 Q/ W9 {2 R2 N. E+ C; I& Z
具体代码如下:

! {5 }* o' K: ?2 n# C* e5 e0 ]1 ^3 ?2 S6 s. i. r0 U

: b& m7 S$ T9 t: T% s
2 C0 N# Y9 w4 p' q5 E( N0 O( {$ p

  s) N3 b$ R3 l  J1 m
1 l& y8 r$ n/ L4 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-4-21 07:00 , Processed in 0.444063 second(s), 55 queries .

回顶部