- 在线时间
- 480 小时
- 最后登录
- 2026-6-1
- 注册时间
- 2023-7-11
- 听众数
- 4
- 收听数
- 0
- 能力
- 0 分
- 体力
- 7823 点
- 威望
- 0 点
- 阅读权限
- 255
- 积分
- 2934
- 相册
- 0
- 日志
- 0
- 记录
- 0
- 帖子
- 1174
- 主题
- 1189
- 精华
- 0
- 分享
- 0
- 好友
- 1
该用户从未签到
 |
一、首先介绍一下0-1背包问题:. c0 X+ ]$ E" D- E# ?) N3 J$ m2 j
0-1背包问题(0-1 Knapsack Problem)是一个经典的组合优化问题,通常在计算机科学和运筹学中讨论。这个问题涉及到一个背包和一组物品,每个物品都有一个特定的重量和价值。问题的目标是在给定背包的最大容量下,选择一组物品放入背包,以使得所选物品的总重量不超过背包的容量,同时最大化这些物品的总价值。
0 s6 f( J: V% j 0-1背包问题的名称中的“0-1”表示每个物品要么完全放入背包(选择)要么完全不放入背包(不选择),不能部分放入。这是问题的一个关键特征,与分数背包问题不同,分数背包问题允许部分放入物品。' S4 n9 n8 C$ a/ _
问题的形式描述如下:
9 Z. j( ~3 ?4 A2 y& F, U 1.给定一个固定容量的背包,通常表示为一个正整数W(背包的最大承载重量)。
+ p& @. z' ^) p- d+ g 2.给定一组物品,每个物品都有两个属性:重量(weight)和价值(value)。
$ S5 ?: V& C; K' X& Q7 } N( k* p 3.对每个物品,你可以选择将其放入背包(选择)或不放入背包(不选择)。
/ [( ~ Q6 `: o9 W 4.每个物品只能选择一次,即要么放入背包,要么不放入。
1 ~4 \8 t: j% o: f 5.目标是选择一个物品组合,使得它们的总重量不超过背包容量W,同时使它们的总价值最大化。: T8 @$ |- a0 w0 c/ l
/ P( W- `6 o* R, R
解决0-1背包问题的一种常见方法是使用动态规划(Dynamic Programming)算法。这个问题有广泛的应用,包括资源分配、排程问题、投资组合优化等领域。它还是计算复杂性理论中的一个经典问题,通常被用来说明NP难问题的概念。
! D5 y4 q0 ]9 i( i- e
3 Q, ]2 g0 d* W" p( L# m! k1 O0 z2 k F
二、 介绍代码
5 _7 N& e6 `1 e6 B$ E5 _$ a- j这段代码是一个Python实现的0-1背包问题的解决方法,使用了动态规划算法来找到最优解。以下是对代码的详细解释:- def knapsack(v, w, n, capacity):
- E$ W6 ?: y, e# ]& o+ R - i = 0' C& I: f o$ H4 F
- capacity = capacity + 1 # 初始化背包容量最大值
/ z\" K \5 s2 P, Q! m, ^ - m = np.zeros((n, capacity)) # 初始化! d: i' y( o5 ^2 X' i4 b' H4 {
- x = np.zeros(n)
复制代码 1.v 是一个长度为 n 的列表,表示每个物品的价值。# {+ _+ p0 R- X2 x0 a; ~& X8 a
2.w 是一个长度为 n 的列表,表示每个物品的重量。
* t& ^6 a: J- ]1 T3.n 表示物品的数量。
9 {1 y8 v0 F6 i4.capacity 表示背包的容量。
- ~2 V$ }8 s. y6 l. X
& U6 Y, J9 Y& Y# ]. z代码首先初始化了一个二维数组 m 作为动态规划表,其中 m[j] 表示在考虑前 i 个物品时,背包容量为 j 时可以获得的最大总价值。数组 x 用来存储最终的解,x 表示是否选择第 i 个物品。- for i in range(n):
8 K# E, I5 N4 d) j2 ]% b! d9 W8 W - for j in range(capacity):
\" X) l( n: Y6 L; z# l: i+ N4 e' O - if (j >= w[i]):% X. {, W& R& W: N) ^4 G
- m[i][j] = max(m[i - 1][j], m[i - 1][j - w[i]] + v[i])9 e0 ~\" d) {) V$ w3 _) U
- else:7 m# Y6 L8 S& ~6 f
- m[i][j] = m[i - 1][j]
复制代码 在这个部分,代码使用了一个嵌套的循环,遍历了所有物品和不同的背包容量。对于每个物品 i 和容量 j,代码计算了两种情况下的最大总价值:) s; j% L9 x* q$ }3 w
5.如果当前物品的重量 w 小于等于当前容量 j,那么可以选择将第 i 个物品放入背包,此时总价值为 m[i-1][j-w] + v,或者选择不放入,此时总价值为 m[i-1][j]。代码选择其中较大的值作为 m[j]。
. D2 x9 `! p' Y0 R6.如果当前物品的重量 w 大于当前容量 j,则无法放入物品,所以总价值等于上一行的值 m[i-1][j]。) e! N5 v" e; B) v( |
. z5 x7 l4 k* Y& x+ ^: m; p这个循环填充了动态规划表 m,最终 m[n-1][capacity] 包含了问题的最优解,即在给定容量下可以获得的最大总价值。- capacity = capacity - 16 x- P7 c7 z: G* h; r5 t' ?% u
- for i in range(n - 1, 0, -1):
+ r: q1 n; e) n% ^4 | - if (m[i][capacity] == m[i - 1][capacity]):
+ |% X) A, d3 f# x\" U% B; z - x[i] = 0
3 l, T7 M' J\" `* y$ Y, [ - else:
( z, A# E2 C. C1 ~ - x[i] = 1% X. W! L. ]! v k
- capacity -= w[i]
8 N# E$ P0 I$ o) f/ z( h - x[0] = 1 if (m[1][capacity] > 0) else 0
复制代码 在这一部分,代码反向遍历动态规划表,从最后一行向前找到解的路径。如果 m[capacity] 等于 m[i-1][capacity],表示第 i 个物品没有放入背包,否则放入背包,并更新剩余容量 capacity。- weight = 0
( O, E/ x# P4 M) u! D- ~ - value = 06 q2 G9 p+ x# y& L( a
- print('装载的物品编号为:')
$ {: Y1 K0 f) A0 Y) G n+ F - for i in range(len(x)):+ q; E$ y/ I% b- v5 d+ G* b! g
- if (x[i] == 1):, q5 ^4 T% B3 C8 l
- weight = weight + w[i]/ _7 l1 [8 S- U7 y8 G- ^
- value = value + v[i]
: M7 A8 D7 ^# C - print(' ', i + 1)( ?\" f T. D j+ z
- print('装载的物品重量为:')
/ A% V- V2 @\" J+ x - print(weight)
! ?$ _$ n* o9 W9 A' o8 w( h - print('装入的物品价值为:')
: B: S1 h' u5 D k( O. d - print(value): N) ]8 X8 t: E/ P2 G* }# a
- return m
复制代码 最后,代码计算了被选择的物品的总重量和总价值,并将它们打印出来。函数返回动态规划表 m。' ]& z! Q4 {6 U
这段代码实现了0-1背包问题的解决方法,它通过动态规划算法找到最优解,即在给定背包容量下可以获得的最大总价值,以及选择哪些物品放入背包。
+ t8 }2 k. L6 i1 X, `5 l* }- {6 K$ d+ U" Y
最后在函数的输入文件中,如下例,第一行物品数量为:5 背包载重量为:10物品的重量列表为:[2, 2, 6, 5, 4] 物品的价值列表为: [6, 3, 5, 4, 6]
6 N3 D+ p/ P: Q6 J, B' v4 a: S2 N
% w! o) J# E O+ X- O! q9 a0 q. j' u5 M1 W( s s* ]) v
接下来展示我们的输出结果: 5 Y0 p/ M k6 `: ?1 J0 T
4 t5 z4 w, z$ T+ R7 o$ U0 [' x' m9 S1 T
! ~+ @! U K& O. J& i具体代码如下: & k% V+ |: I, J: I, m9 A6 B
& o2 [# @8 p3 f
9 E9 Y6 g2 B% q4 ~: ^6 j1 m/ ?" ^ c
$ ^ R1 E# B) n8 ]( K7 J$ Y/ S/ ]8 n
6 R6 L# Q2 k+ V( [
|
zan
|