(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++统计函数执行时间的最佳实践

《C++统计函数执行时间的最佳实践》在软件开发过程中,性能分析是优化程序的重要环节,了解函数的执行时间分布对于识别性能瓶颈至关重要,本文将分享一个C++函数执行时间统计工具,希望对大家有所帮助... 目录前言工具特性核心设计1. 数据结构设计2. 单例模式管理器3. RAII自动计时使用方法基本用法高级用法

MyBatis-plus处理存储json数据过程

《MyBatis-plus处理存储json数据过程》文章介绍MyBatis-Plus3.4.21处理对象与集合的差异:对象可用内置Handler配合autoResultMap,集合需自定义处理器继承F... 目录1、如果是对象2、如果需要转换的是List集合总结对象和集合分两种情况处理,目前我用的MP的版本

C# LiteDB处理时间序列数据的高性能解决方案

《C#LiteDB处理时间序列数据的高性能解决方案》LiteDB作为.NET生态下的轻量级嵌入式NoSQL数据库,一直是时间序列处理的优选方案,本文将为大家大家简单介绍一下LiteDB处理时间序列数... 目录为什么选择LiteDB处理时间序列数据第一章:LiteDB时间序列数据模型设计1.1 核心设计原则

深入解析C++ 中std::map内存管理

《深入解析C++中std::map内存管理》文章详解C++std::map内存管理,指出clear()仅删除元素可能不释放底层内存,建议用swap()与空map交换以彻底释放,针对指针类型需手动de... 目录1️、基本清空std::map2️、使用 swap 彻底释放内存3️、map 中存储指针类型的对象

使用SpringBoot+InfluxDB实现高效数据存储与查询

《使用SpringBoot+InfluxDB实现高效数据存储与查询》InfluxDB是一个开源的时间序列数据库,特别适合处理带有时间戳的监控数据、指标数据等,下面详细介绍如何在SpringBoot项目... 目录1、项目介绍2、 InfluxDB 介绍3、Spring Boot 配置 InfluxDB4、I

C++ STL-string类底层实现过程

《C++STL-string类底层实现过程》本文实现了一个简易的string类,涵盖动态数组存储、深拷贝机制、迭代器支持、容量调整、字符串修改、运算符重载等功能,模拟标准string核心特性,重点强... 目录实现框架一、默认成员函数1.默认构造函数2.构造函数3.拷贝构造函数(重点)4.赋值运算符重载函数

MySQL按时间维度对亿级数据表进行平滑分表

《MySQL按时间维度对亿级数据表进行平滑分表》本文将以一个真实的4亿数据表分表案例为基础,详细介绍如何在不影响线上业务的情况下,完成按时间维度分表的完整过程,感兴趣的小伙伴可以了解一下... 目录引言一、为什么我们需要分表1.1 单表数据量过大的问题1.2 分表方案选型二、分表前的准备工作2.1 数据评估

C++ vector越界问题的完整解决方案

《C++vector越界问题的完整解决方案》在C++开发中,std::vector作为最常用的动态数组容器,其便捷性与性能优势使其成为处理可变长度数据的首选,然而,数组越界访问始终是威胁程序稳定性的... 目录引言一、vector越界的底层原理与危害1.1 越界访问的本质原因1.2 越界访问的实际危害二、基

c++日志库log4cplus快速入门小结

《c++日志库log4cplus快速入门小结》文章浏览阅读1.1w次,点赞9次,收藏44次。本文介绍Log4cplus,一种适用于C++的线程安全日志记录API,提供灵活的日志管理和配置控制。文章涵盖... 目录简介日志等级配置文件使用关于初始化使用示例总结参考资料简介log4j 用于Java,log4c

C++归并排序代码实现示例代码

《C++归并排序代码实现示例代码》归并排序将待排序数组分成两个子数组,分别对这两个子数组进行排序,然后将排序好的子数组合并,得到排序后的数组,:本文主要介绍C++归并排序代码实现的相关资料,需要的... 目录1 算法核心思想2 代码实现3 算法时间复杂度1 算法核心思想归并排序是一种高效的排序方式,需要用