根据需求定义“任务”是一个完整的业务搬运流程,整个流程涉及到多个机构(设备)分别动作执行多个步骤,所以依据前面的模型设计,需要把任务分解到多个连续的子任务(作业),未来通过顺序串联下达执行的方式来分步骤的完成任务的执行。

1.1. 仓库规划

  同样,依据需求我们先来做几个仓库的规划设计,101位置是AGV的入库起始站台,102是提升机门口的入库工位,104是提升机1楼轿厢工位,同理504是提升机在5楼的轿厢工位。左边是5楼的货位区编码规则,完整好的货位号加上楼层编码05-01-01表示五楼的01-01货位,如下图:

  有了图上的基本仓库规划设计,我们才能把一个任务依据设计进行分解,例如:任务编码100的搬运任务是把货物从101 入库工位搬运到05-01-01货位存放。

    设计约定:

    ① 102、502为进提升机工位,103、503为出提升机工位,出和入的工位分开来解决任务冲突的问题(当然也可以先规划一个工位,随着程序迭代改进)。

    ② 提升机控制逻辑与电梯类似,到达指定楼层后会自动开门。

    ③ 电梯只有关上廊门才能提升/下降。

  这样根据约定(前置条件)现在我们针对这个任务逐步分解成以下子任务(作业)如下表,对照早期的需求会发现有增加作业,实际项目中也是如此,早期的需求定义到实际编码的时候,需要根据实际的设备接口协议,规定等调整早期的需求定义。

序号

作业描述

执行设备

1

调度AGV从101站台搬运托盘到102站台

1楼AGV

2

调度提升机到1楼并打开门

提升机

3

调度AGV从102工位搬运到104工位并卸货

1楼AGV

4

调度空AGV从104工位到103工位

1楼AGV

5

调度提升机关门

提升机

6

调度提升机1楼提升到5楼并开门

提升机

7

调度空AGV到504工位并载货

5楼AGV

8

调度AGV从504工位到503工位

5楼AGV

9

调度提升机关门

提升机

10

调度AGV从503工位搬运到05-01-01货位并卸货

5楼AGV

  上面的分解比需求表增加了两个子任务“调度空AGV从104工位到102工位”和“调度AGV从504工位到502工位”,是与实际的AGV设备对接通信协议后增加的,AGV小车的控制系统不能识别“调度AGV从提升机1楼门口工位到提升机并卸货,返回提升机门口工位”需求里“返回”提升机门口工位。依据设备的接口逻辑,只能把这个步骤再分解成两个步骤来执行。

  上述分解提供一个简化分解的模型和思路,实际项目可能更为复杂或简单。

1.2. 分解代码实现

  我们在后台任务管理里添加任务编码100、源地址101、目标地址05-01-01的任务,如下图:

  同时admin.py里增加新增任务默认状态设置为未处理的代码,我们通过pk值是否为None来判断model是否是新增还是修改。

  1. #Task模型的管理器
  2. class TaskAdmin(admin.ModelAdmin):
  3. ...
  4.  
  5. def save_model(self, request, obj, form, change):
  6. #新增任务默认状态设置为 未处理
  7. if obj.pk==None:
  8. obj.State=1
  9. obj.User=request.user
  10. return super().save_model(request, obj, form, change)

  接下来,我们演示如何依据上面的分解逻辑用代码来实现把任务分解成子任务并保存到Job对应的表里,同时,把任务的状态从“未处理”更新到“处理完成”状态。这里需要注意的就是任务的分解和状态变更必须在一个事务里完成,系统不能存在任务状态已经变更到处理完成,但是没有对应的作业,也不能存在已有对应的作业任务状态仍然是未处理状态的情形,否则就会造成系统业务上的混乱,前面章节提到的事务应用就非常重要!

  1. #Task模型的管理器
  2. class TaskAdmin(admin.ModelAdmin):
  3. ...
  4.  
  5. @atomic
  6. def task_decompose_action(self, request, queryset):
  7. for obj in queryset:
  8. #只处理状态等于未处理的任务
  9. if obj.State==1:
  10. result=self.task_decompose(request,obj)
  11. if result:
  12. self.message_user(request, str(obj.TaskNum) + " 处理成功.")
  13. else:
  14. self.message_user(request, str(obj.TaskNum) + " 处理成功.")
  15.  
  16. task_decompose_action.short_description = '处理所选的' + ' 任务'
  17.  
  18. def task_decompose(self,request,obj):
  19. success=True
  20. try:
  21. job1=Job(**{"Task":obj,"TaskNum":obj.TaskNum,"OrderNo":1,"Source":obj.Source,"Target":'102',"Executor":"AGV01","State":1,\
  22. "Priority":obj.Priority,"Barcode":obj.Barcode,"User":request.user,})
  23. job1.save()
  24. job2=Job(**{"Task":obj,"TaskNum":obj.TaskNum,"OrderNo":2,"Source":None,"Target":'1',"Executor":"ELEVATOR","State":1,\
  25. "Priority":obj.Priority,"Barcode":obj.Barcode,"User":request.user,})
  26. job2.save()
  27. job3=Job(**{"Task":obj,"TaskNum":obj.TaskNum,"OrderNo":3,"Source":"102","Target":'104',"Executor":"AGV01","State":1,\
  28. "Priority":obj.Priority,"Barcode":obj.Barcode,"User":request.user,})
  29. job3.save()
  30. job4=Job(**{"Task":obj,"TaskNum":obj.TaskNum,"OrderNo":4,"Source":'104',"Target":'103',"Executor":"AGV01","State":1,\
  31. "Priority":obj.Priority,"Barcode":obj.Barcode,"User":request.user,})
  32. job4.save()
  33. job5=Job(**{"Task":obj,"TaskNum":obj.TaskNum,"OrderNo":5,"Source":'1',"Target":'5',"Executor":"ELEVATOR","State":1,\
  34. "Priority":obj.Priority,"Barcode":obj.Barcode,"User":request.user,})
  35. job5.save()
  36. job6=Job(**{"Task":obj,"TaskNum":obj.TaskNum,"OrderNo":6,"Source":'504',"Target":'503',"Executor":"AGV05","State":1,\
  37. "Priority":obj.Priority,"Barcode":obj.Barcode,"User":request.user,})
  38. job6.save()
  39. job7=Job(**{"Task":obj,"TaskNum":obj.TaskNum,"OrderNo":7,"Source":'5',"Target":None,"Executor":"ELEVATOR","State":1,\
  40. "Priority":obj.Priority,"Barcode":obj.Barcode,"User":request.user,})
  41. job7.save()
  42. job8=Job(**{"Task":obj,"TaskNum":obj.TaskNum,"OrderNo":8,"Source":'503',"Target":'05-01-01',"Executor":"AGV05","State":1,\
  43. "Priority":obj.Priority,"Barcode":obj.Barcode,"User":request.user,})
  44. job8.save()
  45. #子任务分解完成,并提交数据库
  46. #更新任务的状态到“处理完成”
  47. obj.State=5
  48. obj.save()
  49. except Exception:
  50. success = False
  51.  
  52. return success

  执行效果如下图:

1.3. 列表增加“作业数量”

  任务分解完成后,列表除了显示状态变化之外,能够查看作业的分解数量,为此在Task Admin里增加job_count函数用来显示作业数量。

  1. #Task模型的管理器
  2. class TaskAdmin(admin.ModelAdmin):
  3. #listdisplay设置要显示在列表中的字段(TaskId字段是Django模型的主键)
  4. list_display = ('TaskId','TaskNum', 'Source', 'Target', 'Barcode','State','PriorityColor','BeginDate','EndDate','job_count','task_operate',)
  5.  
  6. ...
  7.  
  8. def job_count(self, obj):
  9. return obj.job_set.count()
  10.  
  11. job_count.short_description = '作业数量'

1.4. 编辑页面查看作业数据

  Job模型外键管理Task模型,我就可以采用TabularInline的方式来显示任务的作业信息,通过编辑界面直接查看对应任务的Job列表。

  1. class JobInline(admin.TabularInline):
  2. model = Job
  3. fields = ('TaskNum','OrderNo','Source', 'Target','Executor','State','BeginDate','EndDate',)
  4. extra = 0 #默认所有条目
  5. # 只读的字段
  6. readonly_fields = ('TaskNum','OrderNo','Source', 'Target','Executor','State','BeginDate','EndDate',)
  7.  
  8. def has_delete_permission(self, request, obj=None):
  9. return False #不允许删除
  10. def has_add_permission(self, request, obj=None):
  11. return False #不允许添加
  12.  
  13. #Task模型的管理器
  14. class TaskAdmin(admin.ModelAdmin):
  15.  
  16. #listdisplay设置要显示在列表中的字段(TaskId字段是Django模型的主键)
  17. list_display = ('TaskId','TaskNum', 'Source', 'Target', 'Barcode','State','PriorityColor','BeginDate','EndDate','job_count','task_operate',)
  18. inlines=[JobInline]
  19. fieldsets = (("任务", {'fields': ['TaskNum', ('Source', 'Target'), 'Barcode','Priority','State','BeginDate','EndDate','User']}),)

1.5. 魔法数(mogic number)

  任务的状态赋值和变更我们采用了直接obj.State=1或obj.State=5的直接赋值数字的方式,这个就是程序的mogic number它们有特殊的含义,编程人员得知道这个数字对应的含义,当状态变多时间拉长,编程人员就得去回忆或来回查看数字对应的状态。好的编程习惯就是用常量来代替魔法数字。重构models.py里的Task和Job代码用常量替换mogic number.

  1. class Task(models.Model):
  2. STATE_NEW =1
  3. STATE_PROCESSED=4
  4. STATE_RUNNING=5
  5. STATE_COMPLETED=99
  6. STATE_CANCEL=-1
  7.  
  8. TASK_STATE=((STATE_NEW,u'未处理'),(STATE_PROCESSED,u'处理成功'),(STATE_RUNNING,u'执行中'),(STATE_COMPLETED,u'完成'),(STATE_CANCEL,u'已取消'))
  9.  
  10. TaskId = models.AutoField(u'ID',primary_key=True, db_column='task_id')
  11. TaskNum = models.IntegerField(u'任务号', null=False, db_column='task_num')
  12. Source = models.CharField(u'源地址', null=False, max_length=50, db_column='source')
  13. Target = models.CharField(u'目标地址', null=False, max_length=50, db_column='target')
  14. Barcode = models.CharField(u'容器条码', null=False, max_length=50, db_column='barcode')
  15. State = models.IntegerField(u'状态', choices=TASK_STATE, null=False, db_column='state')
  16. Priority = models.IntegerField(u'优先级', choices=PRIORITY, null=True, db_column='priority')
  17. BeginDate = models.DateTimeField(u'开始时间',null=True, db_column='begin_date')
  18. EndDate = models.DateTimeField(u'结束时间',null=True, db_column='end_date')
  19. SystemDate = models.DateTimeField(u'系统时间', null=False, auto_now_add=True, db_column='system_date')
  20. User = models.ForeignKey(User, verbose_name="操作员",null=True, on_delete=models.CASCADE,db_column='user_id')
  21.  
  22. class Meta:
  23. db_table = 'task_task'
  24. ordering = ['-Priority','TaskId']
  25. verbose_name = verbose_name_plural = "任务"
  26.  
  27. ...
  28.  
  29. class Job(models.Model):
  30.  
  31. STATE_NEW =1
  32. STATE_START=2
  33. STATE_COMPLETED=99
  34. STATE_CANCEL=-1
  35.  
  36. JOB_STATE=((STATE_NEW,u'新作业'),(STATE_START,u'下达执行'), (STATE_COMPLETED,u'完成'),(STATE_CANCEL,u'已取消'))

  凡是有魔法数字的地方都尽量替换成可以阅读的常量,采用中式英语都是可以推荐的方式:)。

  1. class TaskBiz(object):
  2. """description of class"""
  3.  
  4. def task_start(self,obj):
  5. success=False
  6. if obj.State==Task.STATE_PROCESSED:
  7. obj.State=Task.STATE_RUNNING
  8. try:
  9. obj.save()
  10. success = True
  11. except Exception:
  12. success = False
  13. return success
  14.  
  15. ...
  16.  
  17. def task_decompose(self,request,obj):
  18. success=True
  19. try:
  20. job1=Job(**{"Task":obj,"TaskNum":obj.TaskNum,"OrderNo":1,"Source":obj.Source,"Target":'102',"Executor":"AGV01","State":Job.STATE_NEW,\
  21. "Priority":obj.Priority,"Barcode":obj.Barcode,"User":request.user,})
  22. job1.save()
  23. job2=Job(**{"Task":obj,"TaskNum":obj.TaskNum,"OrderNo":2,"Source":None,"Target":'1',"Executor":"ELEVATOR","State":Job.STATE_NEW,\
  24. "Priority":obj.Priority,"Barcode":obj.Barcode,"User":request.user,})
  25. job2.save()
  26. job3=Job(**{"Task":obj,"TaskNum":obj.TaskNum,"OrderNo":3,"Source":"102","Target":'104',"Executor":"AGV01","State":Job.STATE_NEW,\
  27. "Priority":obj.Priority,"Barcode":obj.Barcode,"User":request.user,})
  28. job3.save()
  29. job4=Job(**{"Task":obj,"TaskNum":obj.TaskNum,"OrderNo":4,"Source":'104',"Target":'103',"Executor":"AGV01","State":Job.STATE_NEW,\
  30. "Priority":obj.Priority,"Barcode":obj.Barcode,"User":request.user,})
  31. job4.save()
  32. job5=Job(**{"Task":obj,"TaskNum":obj.TaskNum,"OrderNo":5,"Source":'1',"Target":'5',"Executor":"ELEVATOR","State":Job.STATE_NEW,\
  33. "Priority":obj.Priority,"Barcode":obj.Barcode,"User":request.user,})
  34. job5.save()
  35. job6=Job(**{"Task":obj,"TaskNum":obj.TaskNum,"OrderNo":6,"Source":'504',"Target":'503',"Executor":"AGV05","State":Job.STATE_NEW,\
  36. "Priority":obj.Priority,"Barcode":obj.Barcode,"User":request.user,})
  37. job6.save()
  38. job7=Job(**{"Task":obj,"TaskNum":obj.TaskNum,"OrderNo":7,"Source":'5',"Target":None,"Executor":"ELEVATOR","State":Job.STATE_NEW,\
  39. "Priority":obj.Priority,"Barcode":obj.Barcode,"User":request.user,})
  40. job7.save()
  41. job8=Job(**{"Task":obj,"TaskNum":obj.TaskNum,"OrderNo":8,"Source":'503',"Target":'05-01-01',"Executor":"AGV05","State":Job.STATE_NEW,\
  42. "Priority":obj.Priority,"Barcode":obj.Barcode,"User":request.user,})
  43. job8.save()
  44. #子任务分解完成,并提交数据库
  45. #更新任务的状态到“处理完成”
  46. obj.State=Task.STATE_PROCESSED
  47. obj.save()
  48. except Exception:
  49. success = False
  50.  
  51. return success

1.6. 小结

  本章节我们讲述了如何通过admin.py来快速的完成页面功能的构建,并通过自定义action快速的实现了任务分解功能,并根据业务进展也逐步的完善了查看页面以内联表的方式显示作业详情。随着业务功能的增加admin.py的代码逐步增多和变得复杂,下一章我们演示如何通过功能内聚和重构代码,增加代码可读性和可维护性。

python工业互联网应用实战6—任务分解的更多相关文章

  1. python工业互联网应用实战2—从需求开始

    前言:随着国家工业2025战略的推进,工业互联网发展将会提速,将迎来一个新的发展时期,越来越多的企业开始逐步的把产线自动化,去年年底投产的小米亦庄的智能工厂就是一个热议的新闻.小米/华为智能工厂只能说 ...

  2. python工业互联网应用实战1—SQL与ORM

    从sql到ORM应该说也是编程体系逐步演化的结果,通过类和对象更好的组织开个过程中遇到的各种业务问题,面向对象的解耦和内聚作为一套有效的方法论,对于复杂的企业应用而言确实能够解决实践过程中很多问题. ...

  3. python工业互联网应用实战3—模型层构建

    本章开始我们正式进入到实战项目开发过程,如何从需求分析获得的实体数据转到模型设计中来,变成Django项目中得模型层.当然,第一步还是在VS2019 IDE环境重创建一个工程项目,本文我们把工程名称命 ...

  4. python工业互联网应用实战7—业务层

    本章我们演示代码是如何"进化"的,实战的企业日常开发过程中,系统功能总伴随着业务的不断增加,早期简单的代码慢慢的越来越复杂,敏捷编程中的"禅"--简单设计.快速 ...

  5. python工业互联网应用实战13—基于selenium的功能测试

    本章节我们再来说说测试,单元测试和功能测试.单元测试我们在数据验证章节简单提过了,本章我们进一步如何用单元测试来测试view的功能代码:同时,也涉及一下基于selenium的功能测试做法.笔者过去的项 ...

  6. python工业互联网应用实战15-前后端分离模式1

    我们在13章节里通过监控界面讲了如何使用jquery的动态加载数据写法,通过简单案例来说明了如何实现动态的刷新监控界面的数据,本章我们将演示如何从Django模板加载数据逐步演化到前后端分离的异步数据 ...

  7. python工业互联网应用实战18—前后端分离模式之jquery vs vue

    前面我们分三章来说明了使用django template与jquery的差别,通过jquery如何来实现前后端的分离,同时再9章节使用vue.js 我们浅尝辄止的介绍了JQuery到vue的切换,由于 ...

  8. python工业互联网应用实战3—Django Admin列表

    Django Admin笔者使用下来可以说是Django框架的开发利器,业务model构建完成后,我们就能快速的构建一个增删查改的后台管理框架.对于大量的企业管理业务开发来说,可以快速的构建一个可发布 ...

  9. python工业互联网应用实战5—Django Admin 编辑界面和操作

    1.1. 编辑界面 默认任务的编辑界面,对于model属性包含"choices"会自动显示下来列表供选择,"datetime"数据类型也默认提供时间选择组件,如 ...

随机推荐

  1. 面试时通过volatile关键字,全面展示线程内存模型的能力

    面试时,面试官经常会通过volatile关键字来考核候选人在多线程方面的能力,一旦被问题此类问题,大家可以通过如下的步骤全面这方面的能力.     1 首先通过内存模型说明volatile关键字的作用 ...

  2. Canal:同步mysql增量数据工具,一篇详解核心知识点

    老刘是一名即将找工作的研二学生,写博客一方面是总结大数据开发的知识点,一方面是希望能够帮助伙伴让自学从此不求人.由于老刘是自学大数据开发,博客中肯定会存在一些不足,还希望大家能够批评指正,让我们一起进 ...

  3. const关键字:终于拥有真正的常量声明语句

    本文首发于个人网站:const关键字:终于拥有真正的常量声明语句 你好,今天大叔想和你唠扯唠扯 ES6 新增的关键字 -- const.在说 const 关键字之前,大叔先和你唠唠大叔自己对 cons ...

  4. python_mmdt:一种基于敏感哈希生成特征向量的python库(一)

    概述 python_mmdt是一种基于敏感哈希的特征向量生成工具.核心算法使用C实现,提高程序执行效率.同时使用python进行封装,方便研究人员使用. 本篇幅主要介绍涉及的相关基本内容与使用,相关内 ...

  5. Sentry(v20.12.1) K8S 云原生架构探索,JavaScript Enriching Events(丰富事件信息)

    系列 Sentry-Go SDK 中文实践指南 一起来刷 Sentry For Go 官方文档之 Enriching Events Snuba:Sentry 新的搜索基础设施(基于 ClickHous ...

  6. OpenCV 和 Dlib 人脸识别基础

    00 环境配置 Anaconda 安装 1 下载 https://repo.anaconda.com/archive/ 考虑到兼容性问题,推荐下载Anaconda3-5.2.0版本. 2 安装 3 测 ...

  7. MySQL增删改操作

    增删改操作 增加 看语法 1. 插入完整数据(顺序插入) 语法一: INSERT INTO 表名(字段1,字段2,字段3-字段n) VALUES(值1,值2,值3-值n); #指定字段来插入数据,插入 ...

  8. (04)-Python3之--字典(dict)操作

    1.定义 字典的关键字:dict 字典由多个键和其对应的值构成的 键-值 对组成,每个键值对用冒号 : 分割,每个键值对之间用逗号 , 分割,整个字典包括在花括号 {} 中. {key1:value1 ...

  9. MySQL调优性能监控之performance schema

    一.performance_schema的介绍 performance:性能 schema:图(表)示,以大纲或模型的形式表示计划或理论. MySQL的performance schema 用于监控M ...

  10. 【练习】goroutine chan 通道 总结

    1. fatal error: all goroutines are asleep - deadlock! 所有的协程都休眠了 - 死锁! package mainimport("fmt&q ...