(LBFSP)中间存储有限流水车间的最大完工时间Cmax的2种计算方法(含c++源代码)

本文主要是介绍(LBFSP)中间存储有限流水车间的最大完工时间Cmax的2种计算方法(含c++源代码),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

    查阅文献时,发现中间存储有限的流水车间在计算最大完工时间C_{max}时,常出现两种计算方式,一种用开始时间S_{i, j}计算,另一种用完工时间C_{i, j}计算:

(1) 用开始时间S_{i, j}计算

(p_{i, j}代表工件i在机器j上的加工时间,B_{j}代表机器jj-1之间的缓冲区容量)

(C_{max} = S_{\pi (n), m} + p_{\pi (n), m})

 (2) 用完工时间C_{i, j}计算

(P_{i, j}代表工件i在机器j上的加工时间,B_{j}代表机器jj+1之间的缓冲区容量)

    两种方式计算出来的C_{max}结果是一样的,但要注意的是,第1种方式求出来的矩阵S_{i, j}指的是各工序的最早可以开始加工的时间点, 而第2种方式求出来的矩阵C_{i, j}指的是各工序的最晚可以结束加工的时间点,因此两者画出来的甘特图略有差别,如下图所示: 

    从图中可以看出,按第1种方式求解画出的甘特图,相当于对所有的存在空闲的加工时间块进行了左移操作,而第2种方式则是进行了右移操作;对于不存在空闲的加工时间块,两种求解方式的甘特图是一致的,这也是它们求解出来的C_{max}一致的原因。

    同样的原理可以应用到一般的置换流水车间调度问题(PFSP)C_{max}的求解当中。

附:

(1) 用开始时间S_{i, j}计算LBFSP的源代码

/* 中间存储有限Cmax计算(使用开始时间计算) */
// solution -- 可行解(排列),如 π = {1,2,...,n}
// T -- 加工时间矩阵,T[π(i) - 1][j - 1]即P[π(i)][j]
// buffer -- buffer[i] (0 ≤ i ≤ n-1) 为机器i+1与机器i+2间的缓冲区容量
vector<vector<double>> calculate_makespan_start_time(const vector<int>& solution, const vector<vector<double>>& T, const vector<int>& buffer) {int n = solution.size(), m = T[0].size();vector<vector<double>> S(n, vector<double>(m, 0));  //调度(完工时间)矩阵,size = (n, m),存放各机器上的调度结果// 1. S(π1,1) = 0//S[0][0] = 0;// 2. S(π1,j) = S(π1,j - 1) + P(π1,j - 1)  j = 2, 3, ... , mfor (int j = 1; j < m; j++){S[0][j] = S[0][j - 1] + T[solution[0] - 1][j - 1];}for (int i = 1; i < n; i++) {// j = 0 时// 3. S(πi,1) = S(π(i - 1),1) + P(π(i - 1),1)  i ≤ B1 + 1if (i < buffer[0] + 1) S[i][0] = S[i - 1][0] + T[solution[i - 1] - 1][0];// 4. S(πi,1) = max( S(π(i - 1),1) + P(π(i - 1),1), S[i - B1 - 1][2] )  i > B1 + 1else S[i][0] = max(S[i - 1][0] + T[solution[i - 1] - 1][0], S[i - buffer[0] - 1][1]);for (int j = 1; j < m - 1; j++) {// 5. S(πi,j) = max( S(π(i - 1),j) + P(π(i - 1),j), S(π(i),j - 1) + P(π(i),j - 1) )  i ≤ Bj + 1, 2 ≤ j ≤ m - 1if (i < buffer[j] + 1) S[i][j] = max(S[i - 1][j] + T[solution[i - 1] - 1][j], S[i][j - 1] + T[solution[i] - 1][j - 1]);/* 6. S(πi,j) = max( S(π(i - 1),j) + P(π(i - 1),j), S(π(i),j - 1) + P(π(i),j - 1), S(π(i - Bj - 1), j + 1) )i > Bj + 1, 2 ≤ j ≤ m - 1  */else S[i][j] = max(max(S[i - 1][j] + T[solution[i - 1] - 1][j], S[i][j - 1] + T[solution[i] - 1][j - 1]), S[i - buffer[j] - 1][j + 1]);}// j = m - 1 时// 7. S(πi,m) = max( S(π(i - 1),m) + P(π(i - 1),m), S(π(i),m - 1) + P(π(i),m - 1) )S[i][m - 1] = max(S[i - 1][m - 1] + T[solution[i - 1] - 1][m - 1], S[i][m - 2] + T[solution[i] - 1][m - 2]);}// 如果只求Cmax,返回 S[n - 1][m - 1] + T[solution[i] - 1][j] 即可// 这里求的是开始时间矩阵S(πi,j),用于画调度结果的甘特图return S;
}

(2) 用完工时间C_{i, j}计算LBFSP的源代码

/* 中间存储有限Cmax计算(使用完工时间计算) */
// solution -- 可行解(排列),如 π = {1,2,...,n}
// T -- 加工时间矩阵,T[π(i) - 1][j - 1]即P[π(i)][j]
// buffer -- buffer[i] (0 ≤ i ≤ n-1) 为机器i+1与机器i+2间的缓冲区容量
vector<vector<double>> calculate_makespan(const vector<int>& solution, const vector<vector<double>>& T, const vector<int>& buffer) {int n = solution.size(), m = T[0].size();vector<vector<double>> C(n, vector<double>(m, 0));  //调度(完工时间)矩阵,size = (n, m),存放各机器上的调度结果// 1. C(π1,1) = P(π1,1)C[0][0] = T[solution[0] - 1][0];// 2. C(π1,j) = C(π1,j - 1) + P(π1,j)  2 ≤ j ≤ mfor (int j = 1; j < m; j++){C[0][j] = C[0][j - 1] + T[solution[0] - 1][j];}for (int i = 1; i < n; i++) {// j = 0 时// 3. C(πi,1) = C(π(i - 1), 1) + P(πi, 1)  i ≤ B1 + 1if (i < buffer[0] + 1) C[i][0] = C[i - 1][0] + T[solution[i] - 1][0];// 4. C(πi,1) = max( C(π(i - 1), 1) + P(πi, 1), C(π(i - B1 - 1), 2) )  i > B1 + 1else C[i][0] = max(C[i - 1][0] + T[solution[i] - 1][0], C[i - buffer[0] - 1][1]);for (int j = 1; j < m - 1; j++) {// 5. C(πi,j) = max{C(π(i - 1), j), C(πi, j - 1)} + P(πi, j)  i ≤ Bj + 1, 2 ≤ j ≤ m - 1if (i < buffer[j] + 1) C[i][j] = max(C[i - 1][j], C[i][j - 1]) + T[solution[i] - 1][j];/* 6. C(πi,j) = max(max( C(π(i - 1), j), C(πi, j - 1) ) + P(πi, j), C(π(i - Bj - 1), j + 1) )i > Bj + 1, 2 ≤ j ≤ m - 1  */else C[i][j] = max(max(C[i - 1][j], C[i][j - 1]) + T[solution[i] - 1][j], C[i - buffer[j] - 1][j + 1]);}// j = m - 1 时// 7. C(πi,m) = max( C(π(i - 1), m), C(πi, m - 1) ) + P(πi, m)C[i][m - 1] = max(C[i - 1][m - 1], C[i][m - 2]) + T[solution[i] - 1][m - 1];}// 如果只求Cmax,返回 C[n - 1][m - 1] 即可// 这里求的是完工时间矩阵C(πi,j),用于画调度结果的甘特图return C;
}

参考文献:

[1]    WANG L, ZHANG L, ZHENG D. An effective hybrid genetic algorithm for flow shop scheduling with limited buffers[J]. Computers & Operations Research, 2006,33(10): 2960-2971.

[2]    王鹏飞. 群智能优化算法及在流水车间调度问题中的应用研究[D]. 吉林大学, 2019.

这篇关于(LBFSP)中间存储有限流水车间的最大完工时间Cmax的2种计算方法(含c++源代码)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

C++11范围for初始化列表auto decltype详解

《C++11范围for初始化列表autodecltype详解》C++11引入auto类型推导、decltype类型推断、统一列表初始化、范围for循环及智能指针,提升代码简洁性、类型安全与资源管理效... 目录C++11新特性1. 自动类型推导auto1.1 基本语法2. decltype3. 列表初始化3

C++11右值引用与Lambda表达式的使用

《C++11右值引用与Lambda表达式的使用》C++11引入右值引用,实现移动语义提升性能,支持资源转移与完美转发;同时引入Lambda表达式,简化匿名函数定义,通过捕获列表和参数列表灵活处理变量... 目录C++11新特性右值引用和移动语义左值 / 右值常见的左值和右值移动语义移动构造函数移动复制运算符

Java获取当前时间String类型和Date类型方式

《Java获取当前时间String类型和Date类型方式》:本文主要介绍Java获取当前时间String类型和Date类型方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录Java获取当前时间String和Date类型String类型和Date类型输出结果总结Java获取

Python实现批量提取BLF文件时间戳

《Python实现批量提取BLF文件时间戳》BLF(BinaryLoggingFormat)作为Vector公司推出的CAN总线数据记录格式,被广泛用于存储车辆通信数据,本文将使用Python轻松提取... 目录一、为什么需要批量处理 BLF 文件二、核心代码解析:从文件遍历到数据导出1. 环境准备与依赖库

C++中detach的作用、使用场景及注意事项

《C++中detach的作用、使用场景及注意事项》关于C++中的detach,它主要涉及多线程编程中的线程管理,理解detach的作用、使用场景以及注意事项,对于写出高效、安全的多线程程序至关重要,下... 目录一、什么是join()?它的作用是什么?类比一下:二、join()的作用总结三、join()怎么

C++中全局变量和局部变量的区别

《C++中全局变量和局部变量的区别》本文主要介绍了C++中全局变量和局部变量的区别,全局变量和局部变量在作用域和生命周期上有显著的区别,下面就来介绍一下,感兴趣的可以了解一下... 目录一、全局变量定义生命周期存储位置代码示例输出二、局部变量定义生命周期存储位置代码示例输出三、全局变量和局部变量的区别作用域

C++中assign函数的使用

《C++中assign函数的使用》在C++标准模板库中,std::list等容器都提供了assign成员函数,它比操作符更灵活,支持多种初始化方式,下面就来介绍一下assign的用法,具有一定的参考价... 目录​1.assign的基本功能​​语法​2. 具体用法示例​​​(1) 填充n个相同值​​(2)

SpringBoot3.X 整合 MinIO 存储原生方案

《SpringBoot3.X整合MinIO存储原生方案》本文详细介绍了SpringBoot3.X整合MinIO的原生方案,从环境搭建到核心功能实现,涵盖了文件上传、下载、删除等常用操作,并补充了... 目录SpringBoot3.X整合MinIO存储原生方案:从环境搭建到实战开发一、前言:为什么选择MinI

c++ 类成员变量默认初始值的实现

《c++类成员变量默认初始值的实现》本文主要介绍了c++类成员变量默认初始值,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录C++类成员变量初始化c++类的变量的初始化在C++中,如果使用类成员变量时未给定其初始值,那么它将被

C++中NULL与nullptr的区别小结

《C++中NULL与nullptr的区别小结》本文介绍了C++编程中NULL与nullptr的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编... 目录C++98空值——NULLC++11空值——nullptr区别对比示例 C++98空值——NUL