事件(event)通常用于为流程生命周期中发生的事情建模。事件总是图形化为圆圈。在BPMN 2.0中,有两种主要的事件分类:*捕获(catching)与抛出(throwing)*事件。
定时触发的相关事件,包括定时器启动事件,定时器捕获中间件事件,定时器边界事件
定时器启动事件(timer start event)在指定时间创建流程实例。在流程只需要启动一次,或者流程需要在特定的时间间隔重复启动时,都可以使用。
*请注意:*子流程不能有定时器启动事件。
*请注意:*定时器启动事件,在流程部署的同时就开始计时。不需要调用startProcessInstanceByXXX就会在时间启动。调用startProcessInstanceByXXX时会在定时启动之外额外启动一个流程。
*请注意:*当部署带有定时器启动事件的流程的更新版本时,上一版本的定时器作业会被移除。这是因为通常并不希望旧版本的流程仍然自动启动新的流程实例。
定时器启动事件,用其中有一个钟表图标的圆圈来表示。
案例:
在定时启动的节点设置时间:
可以通过FlowableUI中的部署来演示,通过FlowableUI我们可以看到,没有启动流程实例的情况下,到里23:14:14秒的时候自动帮助我们创建了一个流程实例。
当第一个人工处理完成后,第二个人工处理的任务需要在2022-03-27T23:25:14之后执行
案例:
通过FlowableUI的案例演示我们可以看到后一个任务是在定时时间之后执行的
人工任务1如果在定义的2022-03-27T23:36:14
这个时间之前还没有处理,那么就会触发定时边界事件,从而从人工任务3.
案例
发布启动流程
然后在张三这个位置我们不完成,等到定时到来,达到定时的时间,任务进入到了人工审批三
在定时事件中我们一定要放开如下的配置:
指定计时器在启动前应等待多长的时间,首先一定时器启动事件为例:
可以通过FlowableUI的应用来验证
发布任务后然后我们等待两分钟就可以看到任务到了zhangsan
的位置。或者我们也可以在SpringBoot整合Flowable的项目中添加对应的Controller来处理
@Autowired
private ProcessEngine processEngine;
@GetMapping("/deploy")
public String deploy(){
Deployment deployment = processEngine.getRepositoryService().createDeployment()
.addClasspathResource("test003.bpmn20.xml")
.name("等待定时器启动事件")
.deploy();
return "部署任务成功....";
}
提交请求完成部署操作
生成了对应的Task记录
注意:在事件中一定要开启异步任务,不然相关的事件是不会触发的!
然后来看看中间事件的等待定时器事件案例:
案例由两个自动任务和一个定时器中间事件组成,在定时任务中绑定了两个JavaDelegate的Java类来处理
public class SignalStartOnedelegate implements JavaDelegate {
@Override
public void execute(DelegateExecution execution) {
System.out.println("-------触发了-111-------->"+ LocalDateTime.now().toString());
}
}
public class SignalStartTwodelegate implements JavaDelegate {
@Override
public void execute(DelegateExecution execution) {
System.out.println("-------触发了222--------->"+ LocalDateTime.now().toString());
}
}
然后流程图中的关联为
另一个类似,然后定时器中间事件的等待时间设置是2分钟。我们部署后通过Java代码来演示看看
@Test
public void test01() throws Exception{
Deployment deployment = processEngine.getRepositoryService().createDeployment()
.addClasspathResource("等待定时器中间事件.bpmn20.xml")
.name("等待定时中间事件...")
.deploy();
System.out.println("-----");
}
然后我们需要启动流程实例,之后等待两分钟看效果
/**
* 启动流程实例
*
*/
@Test
public void startProcessInstanceByKey() throws Exception{
processEngine.getRuntimeService()
.startProcessInstanceById("Test04:1:325edb10-ae95-11ec-a77f-c03c59ad2248");
// 需要在此阻塞比等待长的时间
TimeUnit.MINUTES.sleep(3);
}
最后我们来看看边界事件中的等待定时器的处理,案例为:
该案例由一个人工审核+两个自动任务+定时器边界事件组成,自动任务一绑定的JavaDelegate是
public class SignalStartOnedelegate implements JavaDelegate {
@Override
public void execute(DelegateExecution execution) {
System.out.println("-------触发了-111-------->"+ LocalDateTime.now().toString());
}
}
自动任务二绑定的JavaDelegate是
public class SignalStartTwodelegate implements JavaDelegate {
@Override
public void execute(DelegateExecution execution) {
System.out.println("-------触发了222--------->"+ LocalDateTime.now().toString());
}
}
定时器边界事件设置的是30S,也就是如果人工审核在30S还没处理就会触发边界事件:通过代码来演示,部署流程后需要启动流程,然后等待30S看控制台输出
@Test
public void test01() throws Exception{
Deployment deployment = processEngine.getRepositoryService().createDeployment()
.addClasspathResource("等待定时器边界事件.bpmn20.xml")
.name("等待定时器边界事件...")
.deploy();
System.out.println("-----");
}
/**
* 启动流程实例
*
*/
@Test
public void startProcessInstanceByKey() throws Exception{
processEngine.getRuntimeService()
.startProcessInstanceById("test05:1:c46f83bf-ae97-11ec-b055-c03c59ad2248");
System.out.println("开始启动的时间:" + LocalDateTime.now().toString());
// 需要在此阻塞比等待长的时间
TimeUnit.MINUTES.sleep(3);
}
等待控制台输出:
小结:timeDuration在三种定时器的事件中
指定重复周期,可用于周期性启动流程,或者为超期用户任务多次发送提醒,这个元素可以使用两种格式
R3/PT10H/${EndDate}
。 当到达endDate时,应用会停止,并为该任务创建其他作业编写案例来演示:
重复时间设置为 R3PT30S 重复3次,间隔30描述,自动任务绑定的是JavaDelegate
public class SignalStartOnedelegate implements JavaDelegate {
@Override
public void execute(DelegateExecution execution) {
System.out.println("-------触发了-111-------->"+ LocalDateTime.now().toString());
}
}
然后我们部署看效果
@Test
public void test01() throws Exception{
Deployment deployment = processEngine.getRepositoryService().createDeployment()
.addClasspathResource("重复启动事件.bpmn20.xml")
.name("等待定时器边界事件...")
.deploy();
System.out.println("-----");
// 需要在此阻塞比等待长的时间
TimeUnit.MINUTES.sleep(3);
}
然后我们再指定下endDate来看看案例,直接可以在xml中来处理
<timerEventDefinition>
<timeCycle>R3/PT30S/2022-03-28T21:46:11+00:00</timeCycle>
</timerEventDefinition>
此外还可以通过cron表达式来处理:
0 0/5 * * * ?
timeCycle作为中间事件的话,只会执行一次,案例如下
案例中的自动任务一二对应绑定如下的JavaDelegate.
public class SignalStartOnedelegate implements JavaDelegate {
@Override
public void execute(DelegateExecution execution) {
System.out.println("-------触发了-111-------->"+ LocalDateTime.now().toString());
}
}
public class SignalStartTwodelegate implements JavaDelegate {
@Override
public void execute(DelegateExecution execution) {
System.out.println("-------触发了222--------->"+ LocalDateTime.now().toString());
}
}
中间事件的设置为R3/PT30S
循环3次,间隔30秒执行,但是这是中间事件,其实只会执行一次,我来看效果
部署后启动
@Autowired
private ProcessEngine processEngine;
@Test
public void test01() throws Exception{
Deployment deployment = processEngine.getRepositoryService().createDeployment()
.addClasspathResource("循环中间事件.bpmn20.xml")
.name("循环中间事件...")
.deploy();
System.out.println("-----");
}
/**
* 启动流程实例
*
*/
@Test
public void startProcessInstanceByKey() throws Exception{
processEngine.getRuntimeService()
.startProcessInstanceById("cycle-middle-event:1:3456ade8-aea7-11ec-9064-c03c59ad2248");
System.out.println("开始启动的时间:" + LocalDateTime.now().toString());
// 需要在此阻塞比等待长的时间
TimeUnit.MINUTES.sleep(3);
}
在边界事件中,定义了循环条件R3/PT30S
理论上要循环3次,间隔30S,单其实也只会执行一次,来看案例
案例上面的等待时间的是一样的,只是边界事件是30S
然后我们部署启动流程看效果
总结:循环设定