(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

相关文章

Windows下C++使用SQLitede的操作过程

《Windows下C++使用SQLitede的操作过程》本文介绍了Windows下C++使用SQLite的安装配置、CppSQLite库封装优势、核心功能(如数据库连接、事务管理)、跨平台支持及性能优... 目录Windows下C++使用SQLite1、安装2、代码示例CppSQLite:C++轻松操作SQ

C++中RAII资源获取即初始化

《C++中RAII资源获取即初始化》RAII通过构造/析构自动管理资源生命周期,确保安全释放,本文就来介绍一下C++中的RAII技术及其应用,具有一定的参考价值,感兴趣的可以了解一下... 目录一、核心原理与机制二、标准库中的RAII实现三、自定义RAII类设计原则四、常见应用场景1. 内存管理2. 文件操

C++中零拷贝的多种实现方式

《C++中零拷贝的多种实现方式》本文主要介绍了C++中零拷贝的实现示例,旨在在减少数据在内存中的不必要复制,从而提高程序性能、降低内存使用并减少CPU消耗,零拷贝技术通过多种方式实现,下面就来了解一下... 目录一、C++中零拷贝技术的核心概念二、std::string_view 简介三、std::stri

C++高效内存池实现减少动态分配开销的解决方案

《C++高效内存池实现减少动态分配开销的解决方案》C++动态内存分配存在系统调用开销、碎片化和锁竞争等性能问题,内存池通过预分配、分块管理和缓存复用解决这些问题,下面就来了解一下... 目录一、C++内存分配的性能挑战二、内存池技术的核心原理三、主流内存池实现:TCMalloc与Jemalloc1. TCM

C++ 函数 strftime 和时间格式示例详解

《C++函数strftime和时间格式示例详解》strftime是C/C++标准库中用于格式化日期和时间的函数,定义在ctime头文件中,它将tm结构体中的时间信息转换为指定格式的字符串,是处理... 目录C++ 函数 strftipythonme 详解一、函数原型二、功能描述三、格式字符串说明四、返回值五

MySQL存储过程之循环遍历查询的结果集详解

《MySQL存储过程之循环遍历查询的结果集详解》:本文主要介绍MySQL存储过程之循环遍历查询的结果集,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录前言1. 表结构2. 存储过程3. 关于存储过程的SQL补充总结前言近来碰到这样一个问题:在生产上导入的数据发现

从基础到进阶详解Pandas时间数据处理指南

《从基础到进阶详解Pandas时间数据处理指南》Pandas构建了完整的时间数据处理生态,核心由四个基础类构成,Timestamp,DatetimeIndex,Period和Timedelta,下面我... 目录1. 时间数据类型与基础操作1.1 核心时间对象体系1.2 时间数据生成技巧2. 时间索引与数据

C++作用域和标识符查找规则详解

《C++作用域和标识符查找规则详解》在C++中,作用域(Scope)和标识符查找(IdentifierLookup)是理解代码行为的重要概念,本文将详细介绍这些规则,并通过实例来说明它们的工作原理,需... 目录作用域标识符查找规则1. 普通查找(Ordinary Lookup)2. 限定查找(Qualif

C/C++ chrono简单使用场景示例详解

《C/C++chrono简单使用场景示例详解》:本文主要介绍C/C++chrono简单使用场景示例详解,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友... 目录chrono使用场景举例1 输出格式化字符串chrono使用场景China编程举例1 输出格式化字符串示

C++/类与对象/默认成员函数@构造函数的用法

《C++/类与对象/默认成员函数@构造函数的用法》:本文主要介绍C++/类与对象/默认成员函数@构造函数的用法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录名词概念默认成员函数构造函数概念函数特征显示构造函数隐式构造函数总结名词概念默认构造函数:不用传参就可以