基于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

相关文章

好题——hdu2522(小数问题:求1/n的第一个循环节)

好喜欢这题,第一次做小数问题,一开始真心没思路,然后参考了网上的一些资料。 知识点***********************************无限不循环小数即无理数,不能写作两整数之比*****************************(一开始没想到,小学没学好) 此题1/n肯定是一个有限循环小数,了解这些后就能做此题了。 按照除法的机制,用一个函数表示出来就可以了,代码如下

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

购买磨轮平衡机时应该注意什么问题和技巧

在购买磨轮平衡机时,您应该注意以下几个关键点: 平衡精度 平衡精度是衡量平衡机性能的核心指标,直接影响到不平衡量的检测与校准的准确性,从而决定磨轮的振动和噪声水平。高精度的平衡机能显著减少振动和噪声,提高磨削加工的精度。 转速范围 宽广的转速范围意味着平衡机能够处理更多种类的磨轮,适应不同的工作条件和规格要求。 振动监测能力 振动监测能力是评估平衡机性能的重要因素。通过传感器实时监

缓存雪崩问题

缓存雪崩是缓存中大量key失效后当高并发到来时导致大量请求到数据库,瞬间耗尽数据库资源,导致数据库无法使用。 解决方案: 1、使用锁进行控制 2、对同一类型信息的key设置不同的过期时间 3、缓存预热 1. 什么是缓存雪崩 缓存雪崩是指在短时间内,大量缓存数据同时失效,导致所有请求直接涌向数据库,瞬间增加数据库的负载压力,可能导致数据库性能下降甚至崩溃。这种情况往往发生在缓存中大量 k

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)

基于UE5和ROS2的激光雷达+深度RGBD相机小车的仿真指南(五):Blender锥桶建模

前言 本系列教程旨在使用UE5配置一个具备激光雷达+深度摄像机的仿真小车,并使用通过跨平台的方式进行ROS2和UE5仿真的通讯,达到小车自主导航的目的。本教程默认有ROS2导航及其gazebo仿真相关方面基础,Nav2相关的学习教程可以参考本人的其他博客Nav2代价地图实现和原理–Nav2源码解读之CostMap2D(上)-CSDN博客往期教程: 第一期:基于UE5和ROS2的激光雷达+深度RG

pip-tools:打造可重复、可控的 Python 开发环境,解决依赖关系,让代码更稳定

在 Python 开发中,管理依赖关系是一项繁琐且容易出错的任务。手动更新依赖版本、处理冲突、确保一致性等等,都可能让开发者感到头疼。而 pip-tools 为开发者提供了一套稳定可靠的解决方案。 什么是 pip-tools? pip-tools 是一组命令行工具,旨在简化 Python 依赖关系的管理,确保项目环境的稳定性和可重复性。它主要包含两个核心工具:pip-compile 和 pip

【VUE】跨域问题的概念,以及解决方法。

目录 1.跨域概念 2.解决方法 2.1 配置网络请求代理 2.2 使用@CrossOrigin 注解 2.3 通过配置文件实现跨域 2.4 添加 CorsWebFilter 来解决跨域问题 1.跨域概念 跨域问题是由于浏览器实施了同源策略,该策略要求请求的域名、协议和端口必须与提供资源的服务相同。如果不相同,则需要服务器显式地允许这种跨域请求。一般在springbo

【Tools】大模型中的自注意力机制

摇来摇去摇碎点点的金黄 伸手牵来一片梦的霞光 南方的小巷推开多情的门窗 年轻和我们歌唱 摇来摇去摇着温柔的阳光 轻轻托起一件梦的衣裳 古老的都市每天都改变模样                      🎵 方芳《摇太阳》 自注意力机制(Self-Attention)是一种在Transformer等大模型中经常使用的注意力机制。该机制通过对输入序列中的每个元素计算与其他元素之间的相似性,

题目1254:N皇后问题

题目1254:N皇后问题 时间限制:1 秒 内存限制:128 兆 特殊判题:否 题目描述: N皇后问题,即在N*N的方格棋盘内放置了N个皇后,使得它们不相互攻击(即任意2个皇后不允许处在同一排,同一列,也不允许处在同一斜线上。因为皇后可以直走,横走和斜走如下图)。 你的任务是,对于给定的N,求出有多少种合法的放置方法。输出N皇后问题所有不同的摆放情况个数。 输入