基于Or-Tools的指派问题建模求解(PythonAPI)

2023-10-29 01:44

本文主要是介绍基于Or-Tools的指派问题建模求解(PythonAPI),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

基于Or-Tools的指派问题建模求解(PythonAPI)

  • 指派问题(又称为分配问题,assignment problem)
  • 基于Or-Tools的指派问题建模求解(PythonAPI)
    • 导入pywraplp库
    • 数据准备
    • 声明MIP求解器
    • 初始化决策变量
    • 初始化约束条件
    • 目标函数
    • 调用求解器
    • 打印结果
    • 求解结果
    • 完整代码

指派问题(又称为分配问题,assignment problem)

指派问题(又称为分配问题,assignment problem)可以抽象概括为:将n个任务(或物品)分配给m个员工(或背包)的问题。其中,最简单的平衡指派模型是指任务数量和员工数量相等的情形。然而,现实生活中的问题大多是任务数量大于员工数量且员工能力有限的广义指派问题(generalized assignment problem,GAP)。GAP是经典的组合优化问题,许多领域的容量约束问题都可以被抽象为GAP进行求解,如机器调度问题、有容量约束的设施选址问题、供应链问题及车辆路径问题等。

广义指派模型
GAP问题可以描述为:将n个相互独立的任务分配给m个员工,一个任务智能由一个员工来完成,一个员工可以完成多项任务,但员工完成任务的总时间不得超过给定限制。

对于 i = 1 , . . . , m i=1,...,m i=1,...,m j = 1 , . . . , n j=1,...,n j=1,...,n,定义0-1决策变量x_ij=1,表示任务j分配给员工i。令 I = { i ∣ i = 1 , ⋯ , m } I=\{i | i=1,\cdots,m\} I={ii=1,,m}为员工集合, J = { j ∣ j = 1 , ⋯ , n } J=\{j | j=1, \cdots, n\} J={jj=1,,n}为任务集合, b i b_i bi表示员工自身的工作时长限制, r i j r_{ij} rij表示员工 i i i完成任务 j j j需要的时长, c i j c_{ij} cij表示员工 i i i完成任务 j j j所消耗的资源或产生的收益。最终目标函数为成本最小或收益最大,则GAP可表述为
max ⁡ 或 min ⁡ ∑ i ∈ I ∑ j ∈ J c i j x i j \begin{align} \max 或 \,\min \sum_{i \in I} \sum_{j \in J} c_{ij}x_{ij} \end{align} maxminiIjJcijxij s . t . s.t. s.t.
∑ j ∈ J r i j x i j ≤ b i , ∀ i ∈ I ∑ i ∈ I x i j = 1 , ∀ j ∈ J x i j ∈ { 0 , 1 } , ∀ j ∈ J \begin{align} \sum_{j \in J} r_{ij}x_{ij} \leq b_i, \quad \forall i \in I \\ \sum_{i \in I}x_{ij}=1, \quad \forall j \in J \\ x_{ij} \in \{0,1\}, \quad \forall j \in J \end{align} jJrijxijbi,iIiIxij=1,jJxij{0,1},jJ
具有上述形式的整数规划模型被称为广义指派模型。

基于Or-Tools的指派问题建模求解(PythonAPI)

在这个例子中,有5个工人(编号0-4)和4个任务(编号0-3),将工人分配给任务的成本如下表所示:

目标为最小化总成本,约束为每个工人最多完成一个任务,每个任务只能由一个工人完成。这个问题中由于工人数多于任务数,因此有一个工人分不到任务。

导入pywraplp库

from ortools.linear_solver import pywraplp

数据准备

costs = [[90, 80, 75, 70],[35, 85, 55, 65],[125, 95, 90, 95],[45, 110, 95, 115],[50, 100, 90, 100],
]
num_workers = len(costs)  # 工人数量
num_tasks = len(costs[0])  # 任务数量

声明MIP求解器

solver = pywraplp.Solver.CreateSolver("SCIP")

初始化决策变量

x = {}
for i in range(num_workers):for j in range(num_tasks):x[i, j] = solver.IntVar(0, 1, "")

初始化约束条件

# 每个员工至多完成一项任务
for i in range(num_workers):solver.Add(solver.Sum([x[i, j] for j in range(num_tasks)]) <= 1)# 每项任务只能由一个员工完成
for j in range(num_tasks):solver.Add(solver.Sum([x[i, j] for i in range(num_workers)]) == 1)

目标函数

objective_terms = []
for i in range(num_workers):for j in range(num_tasks):objective_terms.append(costs[i][j] * x[i, j])
solver.Minimize(solver.Sum(objective_terms))

调用求解器

status = solver.Solve()

打印结果

if status == pywraplp.Solver.OPTIMAL or status == pywraplp.Solver.FEASIBLE:print(f"Total cost = {solver.Objective().Value()}\n")for i in range(num_workers):for j in range(num_tasks):# Test if x[i,j] is 1 (with tolerance for floating point arithmetic).if x[i, j].solution_value() > 0.5:print(f"Worker {i} assigned to task {j}." + f" Cost: {costs[i][j]}")
else:print("No solution found.")

求解结果

Total cost = 265.0
Worker 0 assigned to task 3. Cost = 70
Worker 1 assigned to task 2. Cost = 55
Worker 2 assigned to task 1. Cost = 95
Worker 3 assigned to task 0. Cost = 45

完整代码

from ortools.linear_solver import pywraplpdef main():# Datacosts = [[90, 80, 75, 70],[35, 85, 55, 65],[125, 95, 90, 95],[45, 110, 95, 115],[50, 100, 90, 100],]num_workers = len(costs)num_tasks = len(costs[0])# Solver# Create the mip solver with the SCIP backend.solver = pywraplp.Solver.CreateSolver("SCIP")if not solver:return# Variables# x[i, j] is an array of 0-1 variables, which will be 1# if worker i is assigned to task j.x = {}for i in range(num_workers):for j in range(num_tasks):x[i, j] = solver.IntVar(0, 1, "")# Constraints# Each worker is assigned to at most 1 task.for i in range(num_workers):solver.Add(solver.Sum([x[i, j] for j in range(num_tasks)]) <= 1)# Each task is assigned to exactly one worker.for j in range(num_tasks):solver.Add(solver.Sum([x[i, j] for i in range(num_workers)]) == 1)# Objectiveobjective_terms = []for i in range(num_workers):for j in range(num_tasks):objective_terms.append(costs[i][j] * x[i, j])solver.Minimize(solver.Sum(objective_terms))# Solvestatus = solver.Solve()# Print solution.if status == pywraplp.Solver.OPTIMAL or status == pywraplp.Solver.FEASIBLE:print(f"Total cost = {solver.Objective().Value()}\n")for i in range(num_workers):for j in range(num_tasks):# Test if x[i,j] is 1 (with tolerance for floating point arithmetic).if x[i, j].solution_value() > 0.5:print(f"Worker {i} assigned to task {j}." + f" Cost: {costs[i][j]}")else:print("No solution found.")if __name__ == "__main__":main()

这篇关于基于Or-Tools的指派问题建模求解(PythonAPI)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/297423

相关文章

MySQL 设置AUTO_INCREMENT 无效的问题解决

《MySQL设置AUTO_INCREMENT无效的问题解决》本文主要介绍了MySQL设置AUTO_INCREMENT无效的问题解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参... 目录快速设置mysql的auto_increment参数一、修改 AUTO_INCREMENT 的值。

关于跨域无效的问题及解决(java后端方案)

《关于跨域无效的问题及解决(java后端方案)》:本文主要介绍关于跨域无效的问题及解决(java后端方案),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录通用后端跨域方法1、@CrossOrigin 注解2、springboot2.0 实现WebMvcConfig

Go语言中泄漏缓冲区的问题解决

《Go语言中泄漏缓冲区的问题解决》缓冲区是一种常见的数据结构,常被用于在不同的并发单元之间传递数据,然而,若缓冲区使用不当,就可能引发泄漏缓冲区问题,本文就来介绍一下问题的解决,感兴趣的可以了解一下... 目录引言泄漏缓冲区的基本概念代码示例:泄漏缓冲区的产生项目场景:Web 服务器中的请求缓冲场景描述代码

Java死锁问题解决方案及示例详解

《Java死锁问题解决方案及示例详解》死锁是指两个或多个线程因争夺资源而相互等待,导致所有线程都无法继续执行的一种状态,本文给大家详细介绍了Java死锁问题解决方案详解及实践样例,需要的朋友可以参考下... 目录1、简述死锁的四个必要条件:2、死锁示例代码3、如何检测死锁?3.1 使用 jstack3.2

解决JSONField、JsonProperty不生效的问题

《解决JSONField、JsonProperty不生效的问题》:本文主要介绍解决JSONField、JsonProperty不生效的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录jsONField、JsonProperty不生效javascript问题排查总结JSONField

github打不开的问题分析及解决

《github打不开的问题分析及解决》:本文主要介绍github打不开的问题分析及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、找到github.com域名解析的ip地址二、找到github.global.ssl.fastly.net网址解析的ip地址三

MySQL版本问题导致项目无法启动问题的解决方案

《MySQL版本问题导致项目无法启动问题的解决方案》本文记录了一次因MySQL版本不一致导致项目启动失败的经历,详细解析了连接错误的原因,并提供了两种解决方案:调整连接字符串禁用SSL或统一MySQL... 目录本地项目启动报错报错原因:解决方案第一个:第二种:容器启动mysql的坑两种修改时区的方法:本地

springboot加载不到nacos配置中心的配置问题处理

《springboot加载不到nacos配置中心的配置问题处理》:本文主要介绍springboot加载不到nacos配置中心的配置问题处理,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录springboot加载不到nacos配置中心的配置两种可能Spring Boot 版本Nacos

Java中JSON格式反序列化为Map且保证存取顺序一致的问题

《Java中JSON格式反序列化为Map且保证存取顺序一致的问题》:本文主要介绍Java中JSON格式反序列化为Map且保证存取顺序一致的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未... 目录背景问题解决方法总结背景做项目涉及两个微服务之间传数据时,需要提供方将Map类型的数据序列化为co

如何解决Druid线程池Cause:java.sql.SQLRecoverableException:IO错误:Socket read timed out的问题

《如何解决Druid线程池Cause:java.sql.SQLRecoverableException:IO错误:Socketreadtimedout的问题》:本文主要介绍解决Druid线程... 目录异常信息触发场景找到版本发布更新的说明从版本更新信息可以看到该默认逻辑已经去除总结异常信息触发场景复