7. 网关
网关类似于程序中的if判断,当满足某一条件时,任务进入下一个指定的节点。 网关有:排他网关、并行网关、包含网关、基于事件网关。
7.1 排他网关
排他网关只会选择一条顺序流执行。 就是说,虽然多个顺序流的条件结果为true,
那么XML中的第一个顺序流(也只有这一条)会被选中,并用来继续运行流程。
如果没有选中任何顺序流,会抛出一个异常。排他网关类似于if-elseif
ProcessDefinition definition = processEngine.getRepositoryService()
.createProcessDefinitionQuery().processDefinitionKey("loan_gateway").latestVersion().singleResult();
Map<String, Object> param1 = new HashMap<>();
param1.put("amount", "5500");
//模拟当前借款单的id,保证task能查出唯一的任务。当然也可以查出一个task list批量处理
String businessKey = System.currentTimeMillis() + "";
//启动任务
ProcessInstance processInstance =
runtimeService.startProcessInstanceByKey("loan_gateway", businessKey, param1);
//查询任务
Task task = taskService.createTaskQuery()
.processDefinitionKey("loan_gateway")
.processInstanceBusinessKey(businessKey).singleResult();
taskService.complete(task.getId());
7.2 并行网关
网关也可以表示流程中的并行情况。最简单的并行网关是 并行网关,它允许将流程 分成多条分支,也可以把多条分支 汇聚到一起。
并行网关的功能是基于进入和外出的顺序流的: ’
• 分支: 并行后的所有外出顺序流,为每个顺序流都创建一个并发分支。
• 汇聚:所有到达并行网关,在此等待的进入分支, 直到所有进入顺序流的分支都到达以后, 流程就会通过汇聚网关。
注意,如果同一个并行网关有多个进入和多个外出顺序流, 它就同时具有分支和汇聚功能。
这时,网关会先汇聚所有进入的顺序流,然后再切分成多个并行分支。 与其他网关的主要区别是,并行网关不会解析条件。
即使顺序流中定义了条件,也会被忽略。
7.3 包含网关
包含网关可以看做是排他网关和并行网关的结合体。 和排他网关一样,你可以在外出顺序流上定义条件,包含网关会解析它们。
但是主要的区别是包含网关可以选择多于一条顺序流,这和并行网关一样。 包含网关的功能是基于进入和外出顺序流的:
• 分支: 所有外出顺序流的条件都会被解析,结果为true的顺序流会以并行方式继续执行, 会为每个顺序流创建一个分支。
• 汇聚: 所有并行分支到达包含网关,会进入等待章台, 直到每个包含流程token的进入顺序流的分支都到达。
这是与并行网关的最大不同。换句话说,包含网关只会等待被选中执行了的进入顺序流。 在汇聚之后,流程会穿过包含网关继续执行。
注意,如果同一个包含节点拥有多个进入和外出顺序流, 它就会同时含有分支和汇聚功能。 这时,网关会先汇聚所有拥有流程token的进入顺序流,
再根据条件判断结果为true的外出顺序流,为它们生成多条并行分支。
8. ServiceTask
定义Service类。
import org.activiti.engine.delegate.DelegateExecution;
import org.activiti.engine.delegate.JavaDelegate;
import java.util.Map;
public class PrinterService implements JavaDelegate {
@Override
public void execute(DelegateExecution delegateExecution) throws Exception {
Map<String, Object> variable = delegateExecution.getVariables();
System.out.println(variable.get("name"));
}
}
启动任务时,PrinterService会自动执行。
Map<String, Object> param1 = new HashMap<>();
param1.put("delegate",new PrinterService());
param1.put("name", "abcd");
param1.put("userId", "123");
ProcessInstance processInstance =
runtimeService.startProcessInstanceByKey("ServiceTask",param1);
9. MailTask邮件任务
使用QQ邮箱测试邮件服务,默认QQ邮箱是不开启的,
<bean id="processEngineConfiguration" class="org.activiti.engine.impl.cfg.StandaloneProcessEngineConfiguration">
<property name="dataSource" ref="dataSource"/>
<property name="databaseSchemaUpdate" value="true"/>
<property name="mailServerHost" value="smtp.qq.com"/>
<property name="mailServerPort" value="465"/>
<property name="mailServerDefaultFrom" value="xxxx@qq.com"/>
<property name="mailServerUsername" value="xxxx@qq.com"/>
<property name="mailServerPassword" value="xxxxx"/>
<property name="mailServerUseSSL" value="true"/>
</bean>
其中的mailServerPassword是QQ邮箱验证成功后生成的授权码