排他网关(ExclusiveGateWay)

Posted Jesai的技术博客

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了排他网关(ExclusiveGateWay)相关的知识,希望对你有一定的参考价值。

网关(ExclusiveGateWay)

作者:Jesai
2018年3月11日 00:13:25

情景:
某一家公司最近在给一个单位做一个财务审批的OA。具体需求是这样的:当部门申请的金额小于一万块的时候,财务部可以直接决策。当部分申请的金额大于一万块并且小于三十万的时候,财务部经理决策,当金额超出了或者等于三十万的时候,需要财务主管来决策。那么我们的工作流应该如何设计呢?
也许有人会说,这不简单,小于一万直接发送到财务部,大于一万小于三十,直接给财务经理。其他就发给财务主管。
你是否注意到,其实这样字可以存在审批漏洞。打个比方,我跟部门一个哥们非常好,我又想申请大于一万的经费。我是不是可以直接越过领导发送给我哥们,让他给我放款。再或者,来了一个新人,新人并不知道这个申请的规则,申请几块钱,也发给了财务主管,财务主管非常生气的说:我很忙的,没空给你处理这些蒜皮小事。
分析:
工作流的作用无非就是自动化,规范化。形成一整套的流程。制定好一些列规则后,能够严格遵守这些规则来处理任务。很多一些国产的工作流很灵活,但是审核机制或者工作机制很不严谨。我是不怎么理解这些工作流中国化的。当然,客户是上帝,客户说了算。我们这里以BPMN2.0的标准。就可以使用排他网关来实现我们的需求。
关于排他网关:
1. 排他网关exclusive gateway 经常使用流程变量决定流程下一步要选择的路径
2. 排他网关(也叫异或网关 XOR gateway,或者更专业的,基于数据的排他网关 exclusive data¬based gateway),用于为流程中的决策建模。
3. 当执行到达这个网关时,所有出口顺序流会按照它们定义的顺序进行计算。条件计算为true的顺序流(当没有设置条件时,认为顺序流定义为true)会被选择用于继续流程。
4. 用排他网关时,只会选择一条顺序流。当多条顺序 流的条件都计算为true时,其中在XML中定义的第一条(也只有这条)会被选择,用于继续流程。如果没有可选的顺序流,会抛出异常。
5. 排他网关,用内部带有’X’图标的标准网关(菱形)表示,\'X’图标代表异或(XOR)的含义。请注意内部没有图标的网关默认为排他网关。BPMN 2.0规范不允许在同一个流程中,混合使用带有及没有X的菱形标志
6. 一个排他网关对应一个以上的顺序流


图1 排他网关

说明:网关是一个菱形◇,排他网关就是菱形里面加个\'X’,\'X’图标代表异或(XOR)的含义



图2 默认顺序流

说明:默认顺序流,用起点带有“斜线”标记的一般顺序流表示。
活动的默认顺序流,由该活动的default属性定义。
当其他连线计算都为false的时候,才会走默认顺序流。

排他网关的实现:

关于资金申请审批流程。
步骤一:设计流程图
我们这里设置一个发起部门,就是资金发起申请部门
流程申请的流程:
1. 小于或者等于一万的,直接走财务审批即可
2. 大于一万小于三十万的,需要财务经理审批。
3. 大于或者等于三十万的,需要爱财务主管审批。



图3 排他网关设计

步骤二:并设置表达式

Activiti里面设置表达式可以用EL(Expression Language)表达式。。表达式语言的灵感来自于 ECMAScript 和 XPath 表达式语言,它提供了在 JSP 中简化表达式的方法,让Jsp的代码更加简化。EL不在本章节的讨论范围内,需要了解的可以自行找资料。
1、语法结构
${expression}
2、[ ]与.运算符
EL 提供“.“和“[ ]“两种运算符来存取数据。
当要存取的属性名称中包含一些特殊字符,如 . 或 - 等并非字母或数字的符号,就一定要使用“[ ]“。例如:
${ user. My-Name}应当改为${user["My-Name"]}
如果要动态取值时,就可以用“[ ]“来做,而“.“无法做到动态取值。例如:
${sessionScope.user[data]}中data 是一个变量

表达式一:${money>1&&money<30}


 


图4 设置表达式

表达式二:${money>=30}


 


图6 设置表达式

无表达式:默认
默认是指其他其他条件都不满足的条件下,走默认路线。



 

图7 设置表达式

流程图代码:

 1 <?xml version=\'1.0\' encoding=\'UTF-8\'?>
 2 <definitions xmlns="http://www.omg.org/spec/BPMN/20100524/MODEL" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:activiti="http://activiti.org/bpmn" xmlns:bpmndi="http://www.omg.org/spec/BPMN/20100524/DI" xmlns:omgdc="http://www.omg.org/spec/DD/20100524/DC" xmlns:omgdi="http://www.omg.org/spec/DD/20100524/DI" typeLanguage="http://www.w3.org/2001/XMLSchema" expressionLanguage="http://www.w3.org/1999/XPath" targetNamespace="http://www.activiti.org/processdef">
 3 <process id="process" isExecutable="true">
 4 <startEvent id="sid-7FF420A9-E0A3-435A-8C64-17F4948E169F" />
 5 <userTask id="sid-77BDF7D7-590C-421B-AE15-8BA17A6C37BB" name="部门" activiti:assignee="admin" />
 6 <exclusiveGateway id="sid-8E4A81E5-2B71-4397-A8F1-242C7B0F4DC9" default="sid-21324A76-AD4C-4B76-989A-4870D2AACA3C" />
 7 <userTask id="sid-E53B2318-C85B-442F-9DA7-56E96341FB79" name="财务经理" activiti:assignee="admin" />
 8 <userTask id="sid-E974CD46-11DC-4860-B277-E452C97D4487" name="财务主管" activiti:assignee="admin" />
 9 <endEvent id="sid-6E6E0074-2729-4217-B273-938DE98CF5BE" />
10 <sequenceFlow id="sid-48D646A3-2C45-4F83-9BAF-0ABECFE842D0" sourceRef="sid-E53B2318-C85B-442F-9DA7-56E96341FB79" targetRef="sid-6E6E0074-2729-4217-B273-938DE98CF5BE" />
11 <sequenceFlow id="sid-B8EA0191-9F35-4D09-BEA6-EF41420A86ED" sourceRef="sid-7FF420A9-E0A3-435A-8C64-17F4948E169F" targetRef="sid-77BDF7D7-590C-421B-AE15-8BA17A6C37BB" />
12 <sequenceFlow id="sid-4298366E-F857-4818-8570-DDF1DA7F41AF" sourceRef="sid-77BDF7D7-590C-421B-AE15-8BA17A6C37BB" targetRef="sid-8E4A81E5-2B71-4397-A8F1-242C7B0F4DC9" />
13 <userTask id="sid-8DE1277A-5CAA-4576-A7CD-55E6272B0A6C" name="财务" activiti:assignee="admin" />
14 <sequenceFlow id="sid-4C05BAAF-2FBC-4220-B832-52FFEED2A855" sourceRef="sid-8DE1277A-5CAA-4576-A7CD-55E6272B0A6C" targetRef="sid-6E6E0074-2729-4217-B273-938DE98CF5BE" />
15 <sequenceFlow id="sid-21324A76-AD4C-4B76-989A-4870D2AACA3C" name="默认" sourceRef="sid-8E4A81E5-2B71-4397-A8F1-242C7B0F4DC9" targetRef="sid-8DE1277A-5CAA-4576-A7CD-55E6272B0A6C" />
16 <sequenceFlow id="sid-8D92C614-D1B7-4AF9-9489-F7AAA9732868" sourceRef="sid-E974CD46-11DC-4860-B277-E452C97D4487" targetRef="sid-6E6E0074-2729-4217-B273-938DE98CF5BE" />
17 <sequenceFlow id="sid-838AEA2C-C494-4830-B78F-20EBA96ECB43" name="金额&lt;30万且大于1万" sourceRef="sid-8E4A81E5-2B71-4397-A8F1-242C7B0F4DC9" targetRef="sid-E53B2318-C85B-442F-9DA7-56E96341FB79">
18 <conditionExpression xsi:type="tFormalExpression"><![CDATA[${money>1&&money<30}]]></conditionExpression>
19 </sequenceFlow>
20 <sequenceFlow id="sid-D916A9B9-DC9A-4FC1-BB6B-377520F5F7FC" name="金额>=30万" sourceRef="sid-8E4A81E5-2B71-4397-A8F1-242C7B0F4DC9" targetRef="sid-E974CD46-11DC-4860-B277-E452C97D4487">
21 <conditionExpression xsi:type="tFormalExpression"><![CDATA[${money>=30}]]></conditionExpression>
22 </sequenceFlow>
23 </process>
24 <bpmndi:BPMNDiagram id="BPMNDiagram_process">
25 <bpmndi:BPMNPlane bpmnElement="process" id="BPMNPlane_process">
26 <bpmndi:BPMNShape bpmnElement="sid-7FF420A9-E0A3-435A-8C64-17F4948E169F" id="BPMNShape_sid-7FF420A9-E0A3-435A-8C64-17F4948E169F">
27 <omgdc:Bounds height="30.0" width="30.0" x="105.0" y="155.0" />
28 </bpmndi:BPMNShape>
29 <bpmndi:BPMNShape bpmnElement="sid-77BDF7D7-590C-421B-AE15-8BA17A6C37BB" id="BPMNShape_sid-77BDF7D7-590C-421B-AE15-8BA17A6C37BB">
30 <omgdc:Bounds height="80.0" width="100.0" x="202.0" y="130.0" />
31 </bpmndi:BPMNShape>
32 <bpmndi:BPMNShape bpmnElement="sid-8E4A81E5-2B71-4397-A8F1-242C7B0F4DC9" id="BPMNShape_sid-8E4A81E5-2B71-4397-A8F1-242C7B0F4DC9">
33 <omgdc:Bounds height="40.0" width="40.0" x="360.0" y="150.0" />
34 </bpmndi:BPMNShape>
35 <bpmndi:BPMNShape bpmnElement="sid-E53B2318-C85B-442F-9DA7-56E96341FB79" id="BPMNShape_sid-E53B2318-C85B-442F-9DA7-56E96341FB79">
36 <omgdc:Bounds height="80.0" width="100.0" x="525.0" y="0.0" />
37 </bpmndi:BPMNShape>
38 <bpmndi:BPMNShape bpmnElement="sid-E974CD46-11DC-4860-B277-E452C97D4487" id="BPMNShape_sid-E974CD46-11DC-4860-B277-E452C97D4487">
39 <omgdc:Bounds height="80.0" width="100.0" x="540.0" y="130.0" />
40 </bpmndi:BPMNShape>
41 <bpmndi:BPMNShape bpmnElement="sid-6E6E0074-2729-4217-B273-938DE98CF5BE" id="BPMNShape_sid-6E6E0074-2729-4217-B273-938DE98CF5BE">
42 <omgdc:Bounds height="28.0" width="28.0" x="735.0" y="156.0" />
43 </bpmndi:BPMNShape>
44 <bpmndi:BPMNShape bpmnElement="sid-8DE1277A-5CAA-4576-A7CD-55E6272B0A6C" id="BPMNShape_sid-8DE1277A-5CAA-4576-A7CD-55E6272B0A6C">
45 <omgdc:Bounds height="80.0" width="100.0" x="529.0" y="254.0" />
46 </bpmndi:BPMNShape>
47 <bpmndi:BPMNEdge bpmnElement="sid-4298366E-F857-4818-8570-DDF1DA7F41AF" id="BPMNEdge_sid-4298366E-F857-4818-8570-DDF1DA7F41AF">
48 <omgdi:waypoint x="302.0" y="170.0" />
49 <omgdi:waypoint x="360.0" y="170.0" />
50 </bpmndi:BPMNEdge>
51 <bpmndi:BPMNEdge bpmnElement="sid-D916A9B9-DC9A-4FC1-BB6B-377520F5F7FC" id="BPMNEdge_sid-D916A9B9-DC9A-4FC1-BB6B-377520F5F7FC">
52 <omgdi:waypoint x="399.54545454545456" y="170.45454545454547" />
53 <omgdi:waypoint x="540.0" y="170.11933174224345" />
54 </bpmndi:BPMNEdge>
55 <bpmndi:BPMNEdge bpmnElement="sid-21324A76-AD4C-4B76-989A-4870D2AACA3C" id="BPMNEdge_sid-21324A76-AD4C-4B76-989A-4870D2AACA3C">
56 <omgdi:waypoint x="392.2127329192547" y="177.78726708074535" />
57 <omgdi:waypoint x="529.0" y="262.89168765743074" />
58 </bpmndi:BPMNEdge>
59 <bpmndi:BPMNEdge bpmnElement="sid-8D92C614-D1B7-4AF9-9489-F7AAA9732868" id="BPMNEdge_sid-8D92C614-D1B7-4AF9-9489-F7AAA9732868">
60 <omgdi:waypoint x="640.0" y="170.0" />
61 <omgdi:waypoint x="735.0" y="170.0" />
62 </bpmndi:BPMNEdge>
63 <bpmndi:BPMNEdge bpmnElement="sid-838AEA2C-C494-4830-B78F-20EBA96ECB43" id="BPMNEdge_sid-838AEA2C-C494-4830-B78F-20EBA96ECB43">
64 <omgdi:waypoint x="392.46923076923076" y="162.46923076923076" />
65 <omgdi:waypoint x="525.0" y="73.54755784061696" />
66 </bpmndi:BPMNEdge>
67 <bpmndi:BPMNEdge bpmnElement="sid-4C05BAAF-2FBC-4220-B832-52FFEED2A855" id="BPMNEdge_sid-4C05BAAF-2FBC-4220-B832-52FFEED2A855">
68 <omgdi:waypoint x="629.0" y="257.52941176470586" />
69 <omgdi:waypoint x="737.6892144673815" y="178.25022003555705" />
70 </bpmndi:BPMNEdge>
71 <bpmndi:BPMNEdge bpmnElement="sid-48D646A3-2C45-4F83-9BAF-0ABECFE842D0" id="BPMNEdge_sid-48D646A3-2C45-4F83-9BAF-0ABECFE842D0">
72 <omgdi:waypoint x="625.0" y="77.35632183908046" />
73 <omgdi:waypoint x="737.7845494056976" y="161.6206403605787" />
74 </bpmndi:BPMNEdge>
75 <bpmndi:BPMNEdge bpmnElement="sid-B8EA0191-9F35-4D09-BEA6-EF41420A86ED" id="BPMNEdge_sid-B8EA0191-9F35-4D09-BEA6-EF41420A86ED">
76 <omgdi:waypoint x="135.0" y="170.0" />
77 <omgdi:waypoint x="202.0" y="170.0" />
78 </bpmndi:BPMNEdge>
79 </bpmndi:BPMNPlane>
80 </bpmndi:BPMNDiagram>
81 </definitions>

 

步骤三:测试

1).如果是大于1万小于30万
完成任务代码

 1 /**完成我的任务*/
 2 @Test
 3 public void completeMyPersonalTask(){
 4 ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();
 5 String taskId="377542";
 6 Map<String,Object> variables=new HashMap<String,Object>();
 7 variables.put("money", 20);
 8 processEngine.getTaskService()//与正在执行的案例管理相关的Service
 9 .complete(taskId,variables);
10 
11 System.out.println("完成任务,任务ID:"+taskId);
12 }

 


 


图8 测试流程跟踪

2).如果是大于或者等于30万

 1 /**完成我的任务*/
 2 @Test
 3 public void completeMyPersonalTask(){
 4 ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();
 5 String taskId="385053";
 6 Map<String,Object> variables=new HashMap<String,Object>();
 7 variables.put("money", 40);
 8 processEngine.getTaskService()//与正在执行的案例管理相关的Service
 9 .complete(taskId,variables);
10 
11 System.out.println("完成任务,任务ID:"+taskId);
12 }

 

 


图9 测试流程跟踪

3).如果是小于1万(没有满足条件的,走默认路径:财务):

 1 /**完成我的任务*/
 2 @Test
 3 public void completeMyPersonalTask(){
 4 ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();
 5 String taskId="385048";
 6 Map<String,Object> variables=new HashMap<String,Object>();
 7 variables.put("money", 0.2);
 8 processEngine.getTaskService()//与正在执行的案例管理相关的Service
 9 .complete(taskId,variables);
10 
11 System.out.println("完成任务,任务ID:"+taskId);
12 }

 


 

图10 测试流程跟踪

4).如果没有设置金额

 1 /**完成我的任务*/
 2 @Test
 3 public void completeMyPersonalTask(){
 4 ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();
 5 String taskId="385048";
 6 //Map<String,Object> variables=new HashMap<String,Object>();
 7 //variables.put("money", 0.2);
 8 processEngine.getTaskService()//与正在执行的案例管理相关的Service
 9 .complete(taskId);
10 
11 System.out.println("完成任务,任务ID:"+taskId);

 

会抛一个异常:Unknown property used in expression: ${money>1&&money<30}

org.activiti.engine.ActivitiException: Unknown property used in expression: ${money>1&&money<30}

5).如果没有默认顺序流,又没有满足的条件(2000)

 



图11 测试流程图设计

 

 1 /**完成我的任务*/
 2 @Test
 3 public void completeMyPersonalTask(){
 4 ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();
 5 String taskId="385022";
 6 Map<String,Object> variables=new HashMap<String,Object>();
 7 variables.put("money", 0.2);
 8 processEngine.getTaskService()//与正在执行的案例管理相关的Service
 9 .complete(taskId,variables);
10 
11 System.out.println("完成任务,任务ID:"+taskId);
12 }

 

会抛异常:No outgoing sequence flow of the exclusive gateway

org.activiti.engine.ActivitiException: No outgoing sequence flow of the exclusive gateway \'sid-8E4A81E5-2B71-4397-A8F1-242C7B0F4DC9\' could be selected for continuing the process

扩展:

假如我部门是确定了,但是我的金额可能是随时变动的,那么有两种方案第一种,全部动态表达式:
财务经理:
${money>num2&&money<num3}
财务主管:
${money>=num1}
财务:
${money<num4}

 1 /**完成我的任务*/
 2 @Test
 3 public void completeMyPersonalTask(){
 4 ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();
 5 String taskId="385063";
 6 Map<String,Object> variables=new HashMap<String,Object>();
 7 variables.put("money", 40);
 8 variables.put("num1", 30);
 9 variables.put("num2", 1);
10 variables.put("num3", 30);
11 variables.put("num3", 1);
12 processEngine.getTaskService()//与正在执行的案例管理相关的Service
13 .complete(taskId,variables);
14 
15 System.out.println("完成任务,任务ID:"+taskId);
16 }

 


 


图12 测试流程跟踪

第二种,后台判断
设置表达式:
财务经理:
${flow==”1” }
财务主管:
${flow==”2” }
财务:
${flow==”3” }

 1 @Test
 2 public void completeMyPersonalTask(){
 3 ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();
 4 String taskId="385072";
 5 Map<String,Object> variables=new HashMap<String,Object>();
 6 int money=40;
 7 if(money>=30){
 8 variables.put("flow", "1");
 9 }else if(money>1&&money<30){
10 variables.put("flow", "2");
11 }else{
12 variables.put("flow", "3");
13 }
14 processEngine.getTaskService()//与正在执行的案例管理相关的Service
15 .complete(taskId,variables);
以上是关于排他网关(ExclusiveGateWay)的主要内容,如果未能解决你的问题,请参考以下文章

Activity并行网关和排他网关

Activiti工作流10.排他网关

Activiti7的使用(六)

第八篇Camunda系列-网关服务

第八篇Camunda系列-网关服务

第十一篇Flowable中的各种网关