- 在线时间
- 479 小时
- 最后登录
- 2026-5-9
- 注册时间
- 2023-7-11
- 听众数
- 4
- 收听数
- 0
- 能力
- 0 分
- 体力
- 7813 点
- 威望
- 0 点
- 阅读权限
- 255
- 积分
- 2931
- 相册
- 0
- 日志
- 0
- 记录
- 0
- 帖子
- 1173
- 主题
- 1188
- 精华
- 0
- 分享
- 0
- 好友
- 1
该用户从未签到
 |
一、首先介绍一下0-1背包问题:
. Y( Q; C/ w% h+ B8 W$ |0 E 0-1背包问题(0-1 Knapsack Problem)是一个经典的组合优化问题,通常在计算机科学和运筹学中讨论。这个问题涉及到一个背包和一组物品,每个物品都有一个特定的重量和价值。问题的目标是在给定背包的最大容量下,选择一组物品放入背包,以使得所选物品的总重量不超过背包的容量,同时最大化这些物品的总价值。
$ F( d) b: i- i 0-1背包问题的名称中的“0-1”表示每个物品要么完全放入背包(选择)要么完全不放入背包(不选择),不能部分放入。这是问题的一个关键特征,与分数背包问题不同,分数背包问题允许部分放入物品。
7 Q8 \6 N& T7 q7 I问题的形式描述如下: 9 _- T/ m& ^9 [1 o5 j/ Y
1.给定一个固定容量的背包,通常表示为一个正整数W(背包的最大承载重量)。
1 |( k0 N2 A2 C0 u" N 2.给定一组物品,每个物品都有两个属性:重量(weight)和价值(value)。! x0 k1 a; c* [2 S3 o p
3.对每个物品,你可以选择将其放入背包(选择)或不放入背包(不选择)。' `# v/ d9 O3 v$ J N+ F/ Y
4.每个物品只能选择一次,即要么放入背包,要么不放入。
- @5 b1 @# G6 b0 A* {( ]" ^ 5.目标是选择一个物品组合,使得它们的总重量不超过背包容量W,同时使它们的总价值最大化。) e; r3 l5 k1 |& A9 T& z$ F r
+ m7 j/ s- W, `解决0-1背包问题的一种常见方法是使用动态规划(Dynamic Programming)算法。这个问题有广泛的应用,包括资源分配、排程问题、投资组合优化等领域。它还是计算复杂性理论中的一个经典问题,通常被用来说明NP难问题的概念。
: G" p# P4 @$ W0 G* h( I# A! Z' w0 ]: i0 y9 w- r4 ?, n* V
8 {" ~* K8 X! f
二、 介绍代码
1 M8 f3 P* C4 k3 [$ `; H$ ], M这段代码是一个Python实现的0-1背包问题的解决方法,使用了动态规划算法来找到最优解。以下是对代码的详细解释:- def knapsack(v, w, n, capacity):. F) l& ?$ J6 o& h) p\" m9 r6 H
- i = 05 W( u8 \% j* o( `4 t' S
- capacity = capacity + 1 # 初始化背包容量最大值! S( @) c8 X \
- m = np.zeros((n, capacity)) # 初始化\" l+ r; _* h2 c% d
- x = np.zeros(n)
复制代码 1.v 是一个长度为 n 的列表,表示每个物品的价值。
% `; W' b, ~. g. }5 i2.w 是一个长度为 n 的列表,表示每个物品的重量。+ ?0 b8 m8 D- H* d- }6 Z
3.n 表示物品的数量。
6 J# h5 W$ p G& s$ L4 V7 K4.capacity 表示背包的容量。8 o: }9 H" ^+ S& @7 n) k# E
/ Y9 ?, n( d* W
代码首先初始化了一个二维数组 m 作为动态规划表,其中 m[j] 表示在考虑前 i 个物品时,背包容量为 j 时可以获得的最大总价值。数组 x 用来存储最终的解,x 表示是否选择第 i 个物品。- for i in range(n):
4 M& E6 H5 N0 L2 s- `. ~9 R/ [ - for j in range(capacity):
X$ K/ Z ]- l, n\" W) N - if (j >= w[i]):& I8 G\" ~$ V9 B
- m[i][j] = max(m[i - 1][j], m[i - 1][j - w[i]] + v[i])2 P* {; c1 R% |6 V6 n3 T7 U3 L
- else:
8 ? q9 q4 L) t. i0 R - m[i][j] = m[i - 1][j]
复制代码 在这个部分,代码使用了一个嵌套的循环,遍历了所有物品和不同的背包容量。对于每个物品 i 和容量 j,代码计算了两种情况下的最大总价值:
8 U" V4 F" Q$ B, o! Y$ d, ]5.如果当前物品的重量 w 小于等于当前容量 j,那么可以选择将第 i 个物品放入背包,此时总价值为 m[i-1][j-w] + v,或者选择不放入,此时总价值为 m[i-1][j]。代码选择其中较大的值作为 m[j]。! M; {% k4 w& l1 n p! o& }/ C% K
6.如果当前物品的重量 w 大于当前容量 j,则无法放入物品,所以总价值等于上一行的值 m[i-1][j]。
2 o U1 _' h# R) U+ d% m! ]- u5 h; y
- \4 k2 `- c9 ^这个循环填充了动态规划表 m,最终 m[n-1][capacity] 包含了问题的最优解,即在给定容量下可以获得的最大总价值。- capacity = capacity - 1- O! J% p' A( d( V\" t0 c
- for i in range(n - 1, 0, -1):0 d1 _; U E8 ^- r+ f
- if (m[i][capacity] == m[i - 1][capacity]):6 t4 A7 a, B7 |, r# h! i$ |2 f
- x[i] = 0
5 _4 D3 y: [, F, r9 [1 ^ - else:, F! I( z5 u& R6 c4 M. _
- x[i] = 1
# p' s( E\" X* g, i- M - capacity -= w[i]: f9 i/ {4 v( ^9 r5 Z5 X
- x[0] = 1 if (m[1][capacity] > 0) else 0
复制代码 在这一部分,代码反向遍历动态规划表,从最后一行向前找到解的路径。如果 m[capacity] 等于 m[i-1][capacity],表示第 i 个物品没有放入背包,否则放入背包,并更新剩余容量 capacity。- weight = 0\" H# a: g! M: Y' p3 g
- value = 0# e+ C( i$ T% H! _
- print('装载的物品编号为:')
0 g# @7 ~: J; S { - for i in range(len(x)):1 W: O( s6 g$ u' h* l- ?/ D. c J' K
- if (x[i] == 1):
( _! w8 q3 f5 y t3 }) ?' E - weight = weight + w[i]4 w1 l, ?! a S: l- ?7 P$ X U: x' h
- value = value + v[i]
9 u0 j' v# P. e1 @4 x\" e& s( m8 l - print(' ', i + 1)) [8 z1 D, j9 M) M6 `
- print('装载的物品重量为:')8 F& G& `) N4 m& j3 Q6 a! I
- print(weight)2 D: U* F/ y% k* G- x+ y
- print('装入的物品价值为:')' `3 G: c q9 E6 o9 w
- print(value)$ X4 G$ L( F1 G; A1 Y# e\" m
- return m
复制代码 最后,代码计算了被选择的物品的总重量和总价值,并将它们打印出来。函数返回动态规划表 m。
* }3 T; r, y( o这段代码实现了0-1背包问题的解决方法,它通过动态规划算法找到最优解,即在给定背包容量下可以获得的最大总价值,以及选择哪些物品放入背包。* G1 _( b2 o b* t
/ p0 A5 z+ b' y. ~4 s
最后在函数的输入文件中,如下例,第一行物品数量为:5 背包载重量为:10物品的重量列表为:[2, 2, 6, 5, 4] 物品的价值列表为: [6, 3, 5, 4, 6]
; f) I! a0 E2 |: G, w5 g
5 T1 E+ T9 b" ?" n s
% ]6 M4 O& k3 I* E接下来展示我们的输出结果: % i6 l% ^; J' Y# i
; o6 E, e) A x6 I( C2 }
% b6 p: R5 j3 J) h0 x/ M
& b& |2 h h3 e! c具体代码如下: ) s3 O- ^- B, j( Q' ?
, P. b5 I5 t" @% v0 W' I
/ N6 F% c4 C- `& F# y0 f5 o8 I4 l4 o9 C5 a' W
& d' w3 t: R* H
6 v, A7 q( w7 ]
|
zan
|