• Flexible-Job-Shop-Scheduling-Problem解读


    关注我的公众号YueTan进行交流探讨
    欢迎关注我的APS仓库:https://github.com/yuetan1988/python-lekin

    APS系列入门

    本次解读的代码来自:

    https://github.com/samy-barrech/Flexible-Job-Shop-Scheduling-Problem/tree/master/app

    关于生产排程的问题。

    点评

    • 方法相对全面
    • 没有体现出图工艺路线,按照线性做下去

    数据

    6   6   1   
    6   1   3   1   1   1   3   1   2   6   1   4   7   1   6   3   1   5   6   
    6   1   2   8   1   3   5   1   5   10  1   6   10  1   1   10  1   4   4   
    6   1   3   5   1   4   4   1   6   8   1   1   9   1   2   1   1   5   7   
    6   1   2   5   1   1   5   1   3   5   1   4   3   1   5   8   1   6   9   
    6   1   3   9   1   2   3   1   5   5   1   6   4   1   1   3   1   4   1   
    6   1   2   3   1   4   3   1   6   9   1   1   10  1   5   4   1   3   1   
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    第一行:代表6个job,6个机器,每个工序平均需要1台机器
    之后的6行代表其中的6job,第二行是第一个job,需要6个工序来完成,第一个工序可以1台机器完成,机器3,需要1个单位时间。

    OOD

    注意从scheduler里记录已完成job, 待完成job; job里记录已完成actitity, 未完成activity; activity里记录已完成op, 待完成op

    • job
      • 待完成activity
      • 已完成activity
    • activity
      • 属于的job
      • 待完成operation
      • 已完成operation
      • 一个工序可以在多个机器完成,时间不一样的话。activity里会同时有多个op
    • operation
      • operation用那个机器
      • 花的时间
    • machine
      • operation
      • 增加operation, 最大operation
      • 处理的operation
    • 排产
      • 待完成job
      • 已完成job

    规则

    1 parse => jobs_list, machines_list

    2 heuristic = Heuristic().select

    3 Scheduler(machine_list, jobs_list)

    4 Scheduler.run(heuristic)

    def run(self, heuristic, verbose=True):
    		# Disable print if verbose is False
    		if not verbose:
    			sys.stdout = None
    
    		current_step = 0
    
    		while len(self.__jobs_to_be_done) > 0:  # 只要有未完成工作
    			current_step += 1
    
    			best_candidates = heuristic(self.__jobs_to_be_done, self.__max_operations, current_step)  # 不同规则返回最佳candidate,一次完成一个工作
    			for id_machine, candidates in best_candidates.items():  # 最佳候选里,是关于多个机器,其能承受最大工序的
    				machine = self.__machines[id_machine - 1]  # id转化为对象
    				for activity, operation in candidates:
    					if not (machine.is_working_at_max_capacity() or activity.is_pending):  # 符合条件的分配给机器
    						machine.add_operation(activity, operation)
    
    			for machine in self.__machines:  # 机器时间开始流动
    				machine.work()
    
    			for job in self.__jobs_to_be_done:  # 检查更新job是否完整状态,更新未完成的job
    				if job.is_done:
    					self.__jobs_to_be_done = list(
    						filter(lambda element: element.id_job != job.id_job, self.__jobs_to_be_done))
    					self.__jobs_done.append(job)
    
    		print(colored("[SCHEDULER]", "green"), "Done in " + str(current_step) + " units of time")
    
    		# Reenable stdout
    		if not verbose:
    			sys.stdout = self.__original_stdout
    
    		return current_step
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33

    对于每个job的排产位于heuristic函数中进行

    class Heuristics:
    	# When a choice between multiple operations is available, always pick the first one
    	@staticmethod
    	def select_first_operation(jobs_to_be_done, max_operations, _):
    		best_candidates = {}
    
    		for job in jobs_to_be_done:  # 从待完成的job里找到best_candidate
    			current_activity = job.current_activity  # 向下选择activity/ operation,由于activity要按顺序完成,选择每个job当前
    			best_operation = current_activity.shortest_operation  # 当前里最短的工序
    
    			if best_candidates.get(best_operation.id_machine) is None:  # 如果最佳候选里暂时没有这台机器的安排
    				best_candidates.update({best_operation.id_machine: [(current_activity, best_operation)]})
    			elif len(best_candidates.get(best_operation.id_machine)) < max_operations:  # 如果最佳候选还能继续安排工作,加上去
    				best_candidates.get(best_operation.id_machine).append((current_activity, best_operation))
    			else:  # 机器已经满载了,那么就从里面选
    				list_operations = best_candidates.get(best_operation.id_machine)  
    
    				for key, (_, operation) in enumerate(list_operations):
    					if operation.duration < best_operation.duration:  # 原本候选集里比最新这个时间更短的,pop一个出去
    						list_operations.pop(key)
    						break
    
    				if len(list_operations) < max_operations:  # 如果有pop,就把最新的加上去。否则其实还是原本的候选
    					list_operations.append((current_activity, best_operation))
    
    		return best_candidates
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26

    时间流动

    亲测

    请添加图片描述

  • 相关阅读:
    2017年某高校848数据结构真题复习
    DAY 12 结构体 共用体 枚举02
    SpringBoot-数据访问配置
    Servlet详解
    服务器防火墙软件 --iptables
    PHP8的类与对象的基本操作之成员方法-PHP8知识详解
    Python基本语法_集合setfrozenset_内建方法详解
    vue3 封装自定义指令,监听元素宽高的变化
    基于Python和mysql开发的智慧校园答题考试系统(源码+数据库+程序配置说明书+程序使用说明书)
    Debezium分享系列之:Debezium2.6稳定版本SQLSerer数据库Debezium connector核心知识点
  • 原文地址:https://blog.csdn.net/weixin_38812492/article/details/128145237