python调用求解器SCIP求解多维背包问题(MKP)

1. 背包问题的分类

背包问题(Knapsack Problem,KP) 是NP完全问题,背包问题可以描述为在充分利用空间的前提下将物品放入背包中使得物品的总价值达到尽可能大。经典的背包问题又根据背包数量、物品是否可以重复选择、约束条件的个数变种成不同的背包问题。

问题分类背包数量物品是否可以被重复选择约束条件
0-1背包问题( 0–1 knapsack problem)一个不可重复一个
多选择背包问题一个可重复但选择次数有限制条件一个
多(重) 背包问题多个不可重复一个且每个背包对应不同的约束条件
多目标背包问题多个不可重复一个
多维背包问题(Multi-dimensional Knapsack Problem)一个不可重复多个

2. 0-1背包问题

Given N N N objects, where the j j j th object owns its weight w j w_j wj and profit p j p_j pj, and a knapsack that can hold a limited weight capability C C C, the goal of this problem is to pack the knapsack so that the objects in it have the maximal value among all possible ways the knapsack can be packed.

2.1 数学模型

在这里插入图片描述

3. 多维背包问题(MKP)

多维背包问题(Multi-dimensional Knapsack Problem, MKP) 又称多约束背包问题(Multi-constrained Knapsack Problem),特点是具有2个或2个以上约束条件。

The multidimensional 0–1 knapsack problem (MKP) is a special case of general linear 0–1 programs. Several names have been mentioned in the literature for the MKP: m-dimensional knapsack problem, multidimensional knapsack problem, multiknapsack problem, multiconstraint 0–1 knapsack problem, etc.

3.1 数学模型

请添加图片描述

  • 参数:
    • m m m: 背包约束的个数;
    • n n n:物品的个数;
    • b i b_i bi: 背包约束 i i i(如重量、体积等);
    • a i j a_{ij} aij:表示物品 j j j对背包的约束 i i i的消耗;
    • c i c_i ci:物品 i i i的价值;
  • 决策变量: x j x_j xj=1 表示物品 j j j放入背包; x j x_j xj=0表示物品 j j j未放入背包

2.2 python代码实现MKP及结果

  • python调用求解器SCIP求解多维背包问题,算例来自OR-Library
import pandas as pd
import numpy as np
import pyscipopt as opt


# ==========测试数据==========
# 背包约束
knapsack_const = {'weight': 600, 'volume': 600}
# 商品信息, 商品有不同的属性,如重量weight、体积volume,且有各自的价值value
items_info = [{'name': 'item_0', 'value': 1898, 'weight': 45, 'volume': 30},
 {'name': 'item_1', 'value': 440, 'weight': 5, 'volume': 20},
 {'name': 'item_2', 'value': 22507, 'weight': 85, 'volume': 125},
 {'name': 'item_3', 'value': 270, 'weight': 150, 'volume': 5},
 {'name': 'item_4', 'value': 14148, 'weight': 65, 'volume': 80},
 {'name': 'item_5', 'value': 3100, 'weight': 95, 'volume': 25},
 {'name': 'item_6', 'value': 4650, 'weight': 30, 'volume': 35},
 {'name': 'item_7', 'value': 30800, 'weight': 12, 'volume': 73},
 {'name': 'item_8', 'value': 615, 'weight': 170, 'volume': 12},
 {'name': 'item_9', 'value': 4975, 'weight': 20, 'volume': 15},
 {'name': 'item_10', 'value': 1160, 'weight': 40, 'volume': 15},
 {'name': 'item_11', 'value': 4225, 'weight': 25, 'volume': 40},
 {'name': 'item_12', 'value': 510, 'weight': 20, 'volume': 5},
 {'name': 'item_13', 'value': 11880, 'weight': 3, 'volume': 10},
 {'name': 'item_14', 'value': 479, 'weight': 7, 'volume': 10},
 {'name': 'item_15', 'value': 440, 'weight': 25, 'volume': 12},
 {'name': 'item_16', 'value': 490, 'weight': 12, 'volume': 10},
 {'name': 'item_17', 'value': 330, 'weight': 22, 'volume': 9},
 {'name': 'item_18', 'value': 110, 'weight': 25, 'volume': 10},
 {'name': 'item_19', 'value': 560, 'weight': 9, 'volume': 20},
 {'name': 'item_20', 'value': 24355, 'weight': 165, 'volume': 60},
 {'name': 'item_21', 'value': 2885, 'weight': 2, 'volume': 40},
 {'name': 'item_22', 'value': 11748, 'weight': 85, 'volume': 50},
 {'name': 'item_23', 'value': 4550, 'weight': 15, 'volume': 36},
 {'name': 'item_24', 'value': 750, 'weight': 9, 'volume': 49},
 {'name': 'item_25', 'value': 3720, 'weight': 2, 'volume': 40},
 {'name': 'item_26', 'value': 1950, 'weight': 4, 'volume': 19},
 {'name': 'item_27', 'value': 10500, 'weight': 100, 'volume': 150}]

model = opt.Model("mkp")

# ==========定义变量==========
# 定义0-1决策变量表示商品是否放入背包中
x = {}
for item in items_info:
    x[item['name']] = model.addVar(vtype="B", name=item['name'])

# ==========定义约束==========
# 对于背包的每个约束,都应该满足
for key in knapsack_const.keys():
    model.addCons(opt.quicksum(item[key] * x[item['name']] for item in items_info) <= knapsack_const[key], "const_" + key)

# ==========定义目标==========
# 最大化放入背包的商品的价值
model.setObjective(opt.quicksum(item['value'] * x[item['name']] for item in items_info), "maximize")
model.optimize()

# ==========输出结果==========
result = []
for item in items_info:
    if model.getVal(x[item['name']]) > 0:
        result.append(x[item['name']])
print('ObjVal=', model.getObjVal())
print('result=', result)
  • 结果
    请添加图片描述

参考文献

[1] Freville A . The multidimensional 0–1 knapsack problem: An overview[J]. European Journal of Operational Research, 2004, 155(1):1-21.

  • 0
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值