「达摩院MindOpt」优化FlowShop流水线作业排班问题

2024-01-13 23:52

本文主要是介绍「达摩院MindOpt」优化FlowShop流水线作业排班问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

FlowShop流水线作业

在企业在面临大量多样化的生产任务时,如何合理地安排流水线作业以提高生产效率及确保交货期成为了一个重要的问题。

一个典型的问题就是FlowShop流水线作业安排问题,也有称为生产下料问题。它涉及到多台机器、多个工序以及多个作业的调度安排。在这个问题中,我们需要对多个作业在一组流水线上的处理顺序进行安排,以使得完成所有作业的总时间最短。

与FlowShop相似的还有JobShop调度问题,它们都涉及到多个作业在多台机器上的处理顺序安排,但FlowShop是顺序安排,处理顺序一致,JobShop是作业处理顺序可以不同,安排更灵活,调度约束相对复杂。通常JobShop调度问题通常比FlowShop调度问题更难求解。

本案例将着重介绍FlowShop问题,将向大家介绍如何运用数学规划方法来解决这一问题,帮助企业优化生产过程,实现生产目标。

1. 问题描述

有一个公司生产几套产品,这些产品各不一样,生产各个环节的耗时会不一致,即在每个工序上的耗时不一致。工厂的机器资源有限,如何合理安排产品的生产,让总生产耗时最短?

image.png

  • 假设所有产品都设计成按同样顺序来经过这些加工工序。
  • 各个产品在各个工序的耗时如下表,当有些工序不需要的时候,我们用耗时0来表示

产品ID产品任务1产品任务2产品任务3产品任务4产品任务5产品任务6产品任务7产品任务8产品任务9产品任务10产品任务11产品任务12产品任务13产品任务14产品任务15产品任务16产品任务17产品任务18产品任务19产品任务20
工序机器15483157177365338278776911429127732876894
工序机器2793119956709960556361737547142186577
工序机器316894915894560235764716341634726757740
工序机器4665831687891135949858593941564054775131
工序机器5585620855335534169138672849478758186828

  • 加工完后,工序资源能立即释放出来去生产下一个产品。(此处时简化问题,把工序准备时间加一起了,实际中工序准备可以分开,同一个产品可以在上一个工序还没有生产完的时候就开始准备,即时间可以有重叠,本案例不讨论此情况)
  • 每个工序同时处理的产品只有一种。

请问,如何安排生产,可以让最后完工的时间最小?

2. 数据准备

根据问题的描述,我们整理相应的数据为方便程序读的数据,如CSV的表格。
另外,这里我们做两份数据,方便大家理解怎么将自己业务数据整理后输入。
数据1:Raw数据:根据原问题的表格整理,prductionTime_Raw.csv文件

产品ID产品任务1产品任务2产品任务3产品任务4产品任务5产品任务6产品任务7产品任务8产品任务9产品任务10产品任务11产品任务12产品任务13产品任务14产品任务15产品任务16产品任务17产品任务18产品任务19产品任务20
工序机器15483157177365338278776911429127732876894
工序机器2793119956709960556361737547142186577
工序机器316894915894560235764716341634726757740
工序机器4665831687891135949858593941564054775131
工序机器5585620855335534169138672849478758186828

数据2:修改后数据,修改内容主要是:

  1. 前一天已经生产到一半的任务,已完成的工序耗时设为0;
  2. 不同产品生产工序有区别,将有些不需要执行的工序,耗时设置为0。

修改后的数据如下,同 prductionTime.csv文件


产品ID产品任务1产品任务2产品任务3产品任务4产品任务5产品任务6产品任务7产品任务8产品任务9产品任务10产品任务11产品任务12产品任务13产品任务14产品任务15产品任务16产品任务17产品任务18产品任务19产品任务20
工序机器1000077365338278776911429127732876894
工序机器20009956709960556061737547142186577
工序机器30049158906023576470634163472675040
工序机器40583168780135949858593941564054775131
工序机器55856208553053416913867284947058186828

3. 数学建模

这是个经典的FlowShop问题,最小化生产时间。决策的变量是不同产品/工件的生产顺序。

数学模型如下:
min ⁡ e M m a x , J m a x s.t. ∑ k ∈ J x i , k = 1 ∀ i ∈ J ∑ i ∈ J x i , k = 1 ∀ k ∈ J e m , k ≥ s m , k + ∑ i ∈ J P i , m ⋅ x i , k ∀ m ∈ M , k ∈ J s m , k + 1 ≥ e m , k ∀ m ∈ M , k ∈ { 1 , 2 , . . , J m a x − 1 } s m + 1 , k ≥ e m , k ∀ k ∈ J , M ∈ { 1 , 2 , . . , M m a x − 1 } s 1 , 1 = 0 s m , k , e m , k ≥ 0 ∀ m ∈ M , k ∈ J x i , k ∈ { 0 , 1 } ∀ k , i ∈ I \begin{array}{rll} \min & e_{{M_{max},J_{max}}} & \\ \text{s.t.} & \sum_{k \in J} x_{i,k} = 1 & \forall i \in J \\ & \sum_{i \in J} x_{i,k} = 1 & \forall k \in J \\ & e_{m,k} \geq s_{m,k} + \sum_{i \in J} P_{i,m} \cdot x_{i,k} & \forall m \in M, k \in J\\ & s_{m,k+1} \geq e_{m,k} & \forall m \in M , k \in \{1,2,..,J_{max}-1\} \\ & s_{m+1,k} \geq e_{m,k} & \forall k \in J , M \in \{1,2,..,M_{max}-1\}\\ & s_{1,1}=0 &\\ & s_{m,k},e_{m,k}\geq 0 & \forall m \in M, k\in J \\ & x_{i,k}\in\{0,1\} & \forall k,i \in I \end{array} mins.t.eMmax,JmaxkJxi,k=1iJxi,k=1em,ksm,k+iJPi,mxi,ksm,k+1em,ksm+1,kem,ks1,1=0sm,k,em,k0xi,k{0,1}iJkJmM,kJmM,k{1,2,..,Jmax1}kJ,M{1,2,..,Mmax1}mM,kJk,iI
更细节的数学公式请查看案例

4. MindOpt APL 建模和求解

MindOpt APL是一款代数建模语言,它可以方便地将数学语言描述成程序,然后调用多种求解器求解。MindOpt Solver支持混合整数线性规划(MILP)问题的求解,可选用它。

改写上面的数据图和数学模型,其中数据为了读取和索引加序号方便,我们把上面的表存储为kkv的长表单形式 prductionTime_long.csvprductionTime_long_Raw.csv,并把产品清单和工序清单名字以数字的形式分别存入:Products_name.csv、Machines_name.csv
(请注意,由于我们里面需要用到m+1这样的顺序概念,因此建议把工序的ID用从1开始的顺序排的数字来记录,编程序更方便。)

代码解析

决策变量
var x[<i,k> in Jobs*Jobs] binary;
  • 这是一个二进制决策变量用于表示作业i是否排在位置k上。
  • 如果作业i排在位置k,则x[i,k]等于1,否则等于0。
  • 注意这里使用了JobsJobs,正常情况下应该是JobsMachines或者其他表示作业与位置之间关系的集合。使用Jobs*Jobs可能是为了表示作业顺序而非作业和机器之间的关系。

var s[<m,k> in Machines * Jobs] >= 0;
  • 这个变量’s’代表作业k在机器m上的开始时间。
  • 约束条件保证’s’的值大于或等于0,表示不能有负的开始时间。

var e[<m,k> in Machines * Jobs] >= 0;
  • 这个变量’e’代表作业k在机器m上的结束时间。
  • 同样地,约束条件保证’e’的值大于或等于0,表示不能有负的结束时间。
决策目标
minimize TotalTime: e[lastMachines,lastJobs];

其中 e[m,k] 表示作业 k 在机器 m 上的结束时间。这里 lastMachines 和 lastJobs 分别指最后一台机器和最后一个作业。

约束
  1. 该约束条件表示每个产品必须在一个位置上进行处理。对于每个产品i,通过对其所有位置k上的处理决策进行求和,得到的结果必须等于1,表示每个产品只能在一个位置上进行处理。
subto constraint_1: forall { i in Jobs} sum {k in Jobs} x[i,k] == 1;
  1. 该约束条件表示每个位置只能处理一个产品。对于每个位置k,通过对其所有产品i的处理决策进行求和,得到的结果必须等于1,表示每个位置只能处理一个产品。
subto constraint_1_1: forall { k in Jobs} sum {i in Jobs} x[i,k] == 1; 
  1. 该约束条件表示产品在每个工序的结束时间必须大于等于产品在该工序的开始时间加上该工序的耗时。对于每个工序m和产品k,通过对产品i在工序m上的处理决策乘以其耗时,求和得到的结果必须小于等于产品在该工序的结束时间。
subto constraint_2: forall {<m,k> in Machines * Jobs} e[m,k] >= s[m,k] + sum {i in Jobs} (ProductionTime[i,m] * x[i,k]); 
  1. 该约束条件表示产品在每个工序的下一个位置上的开始时间必须大于等于产品在当前位置的结束时间。对于每个工序m和产品k,下一个位置k+1上的开始时间必须大于等于当前位置k上的结束时间。
subto constraint_3: forall {<m,k> in Machines * (Jobs without {lastJobs})} s[m,k+1] >= e[m,k];
  1. 该约束条件表示产品在下一个工序的同一位置上的开始时间必须大于等于产品在当前工序的结束时间。对于每个工序m和产品k,下一个工序m+1上的同一位置k的开始时间必须大于等于当前工序m上的结束时间。
subto constraint_3_1: forall {<m,k> in (Machines without {lastMachines}) * Jobs } s[m+1,k] >= e[m,k]; 
  1. 该约束条件表示第一个工序中第一个产品的开始时间必须为0,即从0时刻开始生产第一个产品。
subto constraint_4: s[1,1] == 0;

完整代码

代码第11和12行是切换不同的问题数据。

然后如下代码,在Notebook的cell中运行它,进行建模、求解,并整理存储结果:

clear model;
option modelname flowshop_demo; #定义存储文件名# ----------建模--------Start----
# flowshop_production.mapl# 读取数据
param fileDir = "data/";
set Jobs = {read fileDir+"Products_name.csv" as "<1n>" skip 1}; # 读取产品任务列表
set Machines = {read fileDir+"Machines_name.csv" as "<1n>" skip 1}; # 读取工序机器列表
#两种任务数据,可改注释#符号切换
param ProductionTime[Jobs*Machines] = read fileDir+"prductionTime_long_Raw.csv" as "<1n,2n> 3n" skip 1; #读取每个产品在每个工序的耗时
#param ProductionTime[Jobs*Machines] = read fileDir+"prductionTime_long.csv" as "<1n,2n> 3n" skip 1; #读取每个产品在每个工序的耗时param lastJobs = card(Jobs);
param lastMachines = card(Machines);
print "总共有{}个产品任务,{}个工序机器" % lastJobs,lastMachines;# 声明变量
var x[<i,k> in Jobs*Jobs] binary;
var s[<m,k> in Machines * Jobs] >= 0;
var e[<m,k> in Machines * Jobs] >= 0;# 申明目标
minimize TotalTime: e[lastMachines,lastJobs];# 约束
subto constraint_1:forall { i in Jobs}sum {k in Jobs} x[i,k] == 1;
subto constraint_1_1:forall { k in Jobs}sum {i in Jobs} x[i,k] == 1;subto constraint_2:forall {<m,k> in Machines * Jobs}e[m,k] >= s[m,k] + sum {i in Jobs} (ProductionTime[i,m] * x[i,k]);subto constraint_3:forall {<m,k> in Machines * (Jobs without {lastJobs})}  s[m,k+1] >= e[m,k];
subto constraint_3_1:forall {<m,k> in (Machines without {lastMachines}) * Jobs }s[m+1,k] >= e[m,k];subto constraint_4:s[1,1] == 0;#求解
option solver mindopt;     # (可选)指定求解用的求解器,默认是MindOpt#option mindopt_options 'max_time=600'; #设置求解器参数
#option mindopt_options 'print=0'; #设置求解器输出级别,减少过程打印
solve;         # 求解# 结果打印和检查结果
print "----------------结果打印和存文件--------------";
#display; #打印太多,注释了
print "最小生产耗时 = ",e[lastMachines,lastJobs];

运行上述代码结果如下:

总共有20个产品任务,5个工序机器
Running mindoptampl
wantsol=1
MindOpt Version 1.0.1 (Build date: 20231114)
Copyright (c) 2020-2023 Alibaba Cloud.Start license validation (current time : 28-DEC-2023 20:40:53).
License validation terminated. Time : 0.006sModel summary.- Num. variables     : 600- Num. constraints   : 315- Num. nonzeros      : 3350- Num. integer vars. : 400- Bound range        : [1.0e+00,1.0e+00]- Objective range    : [1.0e+00,1.0e+00]Branch-and-cut method started.
Original model: nrow = 315 ncol = 600 nnz = 3350
Tolerance: primal = 1e-06 int = 1e-06 mipgap = 0.0001 mipgapAbs = 1e-06
Limit: time = 1.79769313486232e+308 node = -1 stalling = -1 solution = -1
presolver terminated; took 0 ms
presolver terminated; took 163 ms
Parallelism: root=4, tree=4accept new sol: obj 1448 bnd vio 1.11022302462516e-16 int vio 1.11022302462516e-16 mipgap 1 time 3
Model summary.- Num. variables     : 476- Num. constraints   : 192- Num. nonzeros      : 7883- Bound range        : [1.0e+00,5.4e+01]- Objective range    : [1.0e+00,8.7e+01]- Matrix range       : [1.0e+00,3.5e+02]Presolver started.
Presolver terminated. Time : 0.002sSimplex method started.
Model fingerprint: =Y2dmN2bgdnYgN2dm5mZIteration       Objective       Dual Inf.     Primal Inf.     Time0     1.45592e+02      0.0000e+00      4.0472e+01     0.00s  595     1.24863e+03      0.0000e+00      0.0000e+00     0.02s  
Postsolver started.
Simplex method terminated. Time : 0.018sRoot relaxation: 1248.62782173302 iteration = 595 time = 0.02#node(P:0 Q:0) #(dual:1248.63 best:1448 gap:13.77%) #time = 3accept new sol: obj 1410 bnd vio 2.8421709430404e-14 int vio 2.22044604925031e-16 mipgap 0.114448353380835 time 3accept new sol: obj 1385 bnd vio 1.95761774137987e-14 int vio 1.45008721583694e-16 mipgap 0.098463666618756 time 3accept new sol: obj 1376 bnd vio 2.22044604925031e-16 int vio 2.22044604925031e-16 mipgap 0.0925669900196054 time 3accept new sol: obj 1349 bnd vio 4.2632564145606e-14 int vio 2.22044604925031e-16 mipgap 0.0744048764025033 time 3accept new sol: obj 1344 bnd vio 2.8421709430404e-13 int vio 1.11022302462516e-16 mipgap 0.0709614421629293 time 4accept new sol: obj 1304 bnd vio 2.8421709430404e-13 int vio 9.99200722162641e-16 mipgap 0.0424633268918535 time 5accept new sol: obj 1291 bnd vio 3.33066907387547e-16 int vio 3.33066907387547e-16 mipgap 0.03282120702322 time 7accept new sol: obj 1290 bnd vio 2.22044604925031e-16 int vio 2.22044604925031e-16 mipgap 0.0320714560209124 time 13accept new sol: obj 1278 bnd vio 2.79866832656529e-14 int vio 3.33066907387547e-16 mipgap 0.0223161708476798 time 24
Branch-and-cut method terminated. Time : 31.571sOPTIMAL; objective 1278.00Completed.
----------------结果打印和存文件--------------
最小生产耗时 = 1278
产品,序号,……,|第1个工序s,耗时,|第2个工序s,耗时,|第3个工序s,耗时,|第4个工序s,耗时,|第5个工序s,耗时,|最后结束时间
1,9,……,|343,54,|397,79,|488,16,|562,66,|645,58,|703
2,8,……,|260,83,|394,3,|399,89,|504,58,|589,56,|645
3,11,……,|474,15,|532,11,|636,49,|714,31,|759,20,|779
4,7,……,|189,71,|285,99,|384,15,|436,68,|504,85,|589
5,10,……,|397,77,|476,56,|532,89,|628,78,|706,53,|759
6,4,……,|71,36,|129,70,|199,45,|255,91,|362,35,|397
7,12,……,|489,53,|543,99,|685,60,|745,13,|779,53,|832
8,15,……,|647,38,|722,60,|799,23,|884,59,|1001,41,|1042
9,2,……,|32,27,|74,5,|79,57,|150,49,|199,69,|268
10,18,……,|849,87,|942,56,|998,64,|1062,85,|1147,13,|1160
11,13,……,|542,76,|642,3,|751,7,|758,85,|866,86,|952
12,20,……,|1030,91,|1135,61,|1196,1,|1197,9,|1206,72,|1278
13,6,……,|175,14,|212,73,|321,63,|397,39,|496,8,|504
14,14,……,|618,29,|647,75,|758,41,|843,41,|952,49,|1001
15,3,……,|59,12,|82,47,|136,63,|199,56,|268,47,|315
16,17,……,|772,77,|928,14,|951,47,|1020,40,|1060,87,|1147
17,1,……,|0,32,|32,21,|53,26,|79,54,|133,58,|191
18,16,……,|685,87,|782,86,|868,75,|943,77,|1042,18,|1060
19,5,……,|107,68,|207,5,|244,77,|346,51,|397,68,|465
20,19,……,|936,94,|1030,77,|1107,40,|1147,31,|1178,28,|1206

这篇关于「达摩院MindOpt」优化FlowShop流水线作业排班问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

关于@MapperScan和@ComponentScan的使用问题

《关于@MapperScan和@ComponentScan的使用问题》文章介绍了在使用`@MapperScan`和`@ComponentScan`时可能会遇到的包扫描冲突问题,并提供了解决方法,同时,... 目录@MapperScan和@ComponentScan的使用问题报错如下原因解决办法课外拓展总结@

MybatisGenerator文件生成不出对应文件的问题

《MybatisGenerator文件生成不出对应文件的问题》本文介绍了使用MybatisGenerator生成文件时遇到的问题及解决方法,主要步骤包括检查目标表是否存在、是否能连接到数据库、配置生成... 目录MyBATisGenerator 文件生成不出对应文件先在项目结构里引入“targetProje

C#使用HttpClient进行Post请求出现超时问题的解决及优化

《C#使用HttpClient进行Post请求出现超时问题的解决及优化》最近我的控制台程序发现有时候总是出现请求超时等问题,通常好几分钟最多只有3-4个请求,在使用apipost发现并发10个5分钟也... 目录优化结论单例HttpClient连接池耗尽和并发并发异步最终优化后优化结论我直接上优化结论吧,

Java内存泄漏问题的排查、优化与最佳实践

《Java内存泄漏问题的排查、优化与最佳实践》在Java开发中,内存泄漏是一个常见且令人头疼的问题,内存泄漏指的是程序在运行过程中,已经不再使用的对象没有被及时释放,从而导致内存占用不断增加,最终... 目录引言1. 什么是内存泄漏?常见的内存泄漏情况2. 如何排查 Java 中的内存泄漏?2.1 使用 J

numpy求解线性代数相关问题

《numpy求解线性代数相关问题》本文主要介绍了numpy求解线性代数相关问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 在numpy中有numpy.array类型和numpy.mat类型,前者是数组类型,后者是矩阵类型。数组

解决systemctl reload nginx重启Nginx服务报错:Job for nginx.service invalid问题

《解决systemctlreloadnginx重启Nginx服务报错:Jobfornginx.serviceinvalid问题》文章描述了通过`systemctlstatusnginx.se... 目录systemctl reload nginx重启Nginx服务报错:Job for nginx.javas

Redis缓存问题与缓存更新机制详解

《Redis缓存问题与缓存更新机制详解》本文主要介绍了缓存问题及其解决方案,包括缓存穿透、缓存击穿、缓存雪崩等问题的成因以及相应的预防和解决方法,同时,还详细探讨了缓存更新机制,包括不同情况下的缓存更... 目录一、缓存问题1.1 缓存穿透1.1.1 问题来源1.1.2 解决方案1.2 缓存击穿1.2.1

vue解决子组件样式覆盖问题scoped deep

《vue解决子组件样式覆盖问题scopeddeep》文章主要介绍了在Vue项目中处理全局样式和局部样式的方法,包括使用scoped属性和深度选择器(/deep/)来覆盖子组件的样式,作者建议所有组件... 目录前言scoped分析deep分析使用总结所有组件必须加scoped父组件覆盖子组件使用deep前言

解决Cron定时任务中Pytest脚本无法发送邮件的问题

《解决Cron定时任务中Pytest脚本无法发送邮件的问题》文章探讨解决在Cron定时任务中运行Pytest脚本时邮件发送失败的问题,先优化环境变量,再检查Pytest邮件配置,接着配置文件确保SMT... 目录引言1. 环境变量优化:确保Cron任务可以正确执行解决方案:1.1. 创建一个脚本1.2. 修

Python 标准库time时间的访问和转换问题小结

《Python标准库time时间的访问和转换问题小结》time模块为Python提供了处理时间和日期的多种功能,适用于多种与时间相关的场景,包括获取当前时间、格式化时间、暂停程序执行、计算程序运行时... 目录模块介绍使用场景主要类主要函数 - time()- sleep()- localtime()- g