本文节选自《疯狂Workflow讲义(第2版)》
京东购买地址:https://item.jd.com/12246565.html
工作流Activiti6电子书:http://blog.csdn.net/boxiong86/article/details/78488562
工作流Activiti6教学视频:http://blog.csdn.net/boxiong86/article/details/78608585
开始事件表示流程的开启,可以使用各种类型的开始事件来启动流程,例如使用定时器开始事件,定时启动业务流程,可以使用错误开始事件来表示错误业务流程的开始,根据前面章节所述,所有的开始事件都是Catching事件,即全部的开始事件都会等待着被触发。
不为开始事件指定任何的触发条件(触发器)的事件为无指定开始事件,使用无指定开始事件,流程引擎并不知道流程将会在什么时候开始,如果需要启动流程,就必须使用RuntimeService的startProcessByXXX方法。需要注意的是,子流程(Sub-Process)中总会是一个无指定开始事件,即使将子流程中的开始事件强制定义为其他开始事件,也会被看作无指定开始事件,因为流程到达子流程(Sub-Process)时,就意味着子流程需要启动,而并不需要其他的启动条件。图11-1为无指定开始事件的图形,代码清单11-5为一个含有无指定开始事件的流程XML配置。
图11-1无指定开始事件图形
代码清单11-5:codes\11\11.3\start-event\resource\bpmn\NoneStartEvent.bpmn
targetRef="usertask1">
targetRef="endevent1">
代码清单11-5的粗体字代码,使用startEvent元素定义了一个开始事件,该元素下没有任何的子元素,表示这个开始事件没有任何的事件定义,是一个无指定开始事件。定义了流程后,要启动该流程,需要使用RuntimeService的startProcessByXXX方法,以下为该流程的启动代码:
//创建流程引擎
ProcessEngine engine = ProcessEngines.getDefaultProcessEngine();
//得到流程存储服务组件
RepositoryService repositoryService = engine.getRepositoryService();
//得到运行时服务组件
RuntimeService runtimeService = engine.getRuntimeService();
//部署流程文件
repositoryService.createDeployment()
.addClasspathResource("bpmn/NoneStartEvent.bpmn").deploy();
runtimeService.startProcessInstanceByKey("myProcess");
在开始事件中加入定时器事件定义,该开始事件成为一个定时器开始事件,当符合时间条件后,流程启动,而并不需要像无指定开始事件一样,需要使用API启动流程。在日常生活中有许多需要定时启动的流程,例如要求项目经理每天下班时检查成员的工作日志,又如需要定时检查服务器端口是否存在等,此时可以使用定时器开始事件来实现流程的定时启动。图11-2定义了一个简单的工作流程,代码清单11-6为该流程的配置。
图11-2定时器开始流程
代码清单11-6:codes\11\11.3\start-event\resource\bpmn\TimerStartEvent.bpmn
targetRef="usertask1">
targetRef="endevent1">>
代码清单11-6中,为开始事件添加了定时器事件定义,并且使用了timeCycle元素,该元素支持使用cron表达式,本例为了能看到测试效果,cron表达式中设置了流程将会在每分钟的第0秒开始,每隔5秒将会启动一次流程,代码清单11-7为运行代码。
代码清单11-7:codes\11\11.3\start-event\src\org\crazyit\activiti\TimerStartEvent.java
//创建流程引擎
ProcessEngine engine = ProcessEngines.getDefaultProcessEngine();
//得到流程存储服务组件
RepositoryService repositoryService = engine.getRepositoryService();
//得到运行时服务组件
RuntimeService runtimeService = engine.getRuntimeService();
//部署流程文件
repositoryService.createDeployment()
.addClasspathResource("bpmn/TimerStartEvent.bpmn").deploy();
//等待时间条件
Thread.sleep(70 * 1000);
//查询流程实例
List
System.out.println(ints.size());
代码清单11-7中,并没有使用启动流程的API,在等待70秒后进行流程实例查询,根据查询的流程实例可知,在等待过程中,定时器已经帮我们启动了若干流程实例。
BPMN2.0规范中规定了定时器开始事件可以使用在最高级流程(Top-Level Process)和事件子流程中(Event Sub-Process),而不能使用在其他子流程(嵌套子流程和调用子流程)中,当前Activiti也不支持定时器事件使用在事件子流程。
为开始事件加入消息事件的定义可以让其成为消息开始事件,此时可以使用RuntimeService的startProcessByMessage方法启动流程。代码清单11-8定义了一个含有消息开始事件的流程,图11-3为流程图。
图11-3消息开始事件
代码清单11-8:codes\11\11.3\start-event\resource\bpmn\MessageStartEvent.bpmn
targetRef="usertask1">
targetRef="endevent1">
代码清单11-8的粗体字代码,定义了一个消息开始事件,其中消息事件的定义引用了“msgA”的消息,那么此时可以使用RuntimeService的startProcessByMessage方法来启动流程,需要注意的是,消息事件定义引用的是“message”元素的id,而startProcessByMessage方法传入的参数是“message”元素的name属性。
在BPMN2.0规范中,消息表示的是流程参与者的沟通信息对象,在一般流程中,流程的各个角色的沟通信息,均会有可能导致流程的开始,流程开始事件,可以理解为另外一种启动流程的方式或者途径,使用该开始事件,即达到“接收消息”的条件后启动流程。代码清单11-9加载代码清单11-8的流程文件并启动流程。
代码清单11-9:codes\11\11.3\start-event\src\org\crazyit\activiti\MessageStartEvent.java
//创建流程引擎
ProcessEngine engine = ProcessEngines.getDefaultProcessEngine();
//得到流程存储服务组件
RepositoryService repositoryService = engine.getRepositoryService();
//得到运行时服务组件
RuntimeService runtimeService = engine.getRuntimeService();
//部署流程文件
repositoryService.createDeployment()
.addClasspathResource("bpmn/MessageStartEvent.bpmn").deploy();
//启动流程
runtimeService.startProcessInstanceByMessage("msgA");
//查询流程
System.out.println("流程实例数量:" + runtimeService.createProcessInstanceQuery().count());
运行代码清单11-9,查询到的流程实例数量为1。
BPMN2.0规定了错误开始事件只能使用在事件子流程(Event Sub-Process)中,该事件不能使用在其他的流程中,包括最高级流程(Top-Level Porcess)、嵌套子流程(Sub-Process)和调用子流程(Call Activity)。假设当前有一个检查服务器8080端口的流程,当流程启动时,会执行检查服务器的8080端口是否存在,如果不存在,则进行事件子流程,该流程对应的流程图如图11-4所示,对应的流程文件内容如代码清单11-10所示。
图11-4含有事件子流程的流程
代码清单11-10:codes\11\11.3\start-event\resource\bpmn\ErrorStartEvent.bpmn
triggeredByEvent="true">
activiti:class="org.crazyit.activiti.HandleErrorDelegate">
targetRef="usertask1">
targetRef="endevent1">
activiti:class="org.crazyit.activiti.CheckServerDelegate">
targetRef="servicetask1">
targetRef="endevent2">
图11-4中,定义了一个普通的流程,该流程中除了开始事件和结束事件外,还有一个Service Task,该ServiceTask对应的是代码清单11-10中的粗体字代码,ServiceTask对应的是CheckServerDelegate类,该类主要用于检查服务器的8080端口是否存在,如果8080端口不存在,则抛出异常,然后错误开始事件捕获到该异常后,就会启动事件子流程。对应的CheckServerDelegate代码如代码清单11-11所示。
代码清单11-11:codes\11\11.3\start-event\src\org\crazyit\activiti\CheckServerDelegate.java
try {
System.out.println("开始检查8080端口");
//连接本机的8080端口
Socket socket = new Socket("127.0.0.1", 8080);
System.out.println("检查8080端口完成");
} catch (Exception e) {
System.out.println("检查时出现异常,抛出错误");
//连接出现异常,则抛出BpmnError,且error code为“error”
throw new org.activiti.engine.delegate.BpmnError("error");
}
代码清单11-11中,使用Socket连接到本机的8080端口,如果连接失败,则为抛出BpmnError,BpmnError是一个RuntimeException,该类会维护一个errorCode的属性,如果设置该属性并抛出,就会触发引用了相同errorCode的错误开始事件,如果一个错误开始事件并没有引用错误(error元素),那么将会不管抛出的errorCode是什么,都会触发该事件。
代码清单11-10中,一个事件子流程中使用了错误开始事件,该事件引用了id为“connectError”的error元素,该error元素的errorCode为“error”,当CheckServerDelegate中抛errorCode为“error”的BpmnError时,就会触发该错误开始事件。本例中为了能看到效果,在子流程中,触发了错误开始事件后,会到达一个ServiceTask,该ServiceTask对应的类是HandleErrorDelegate(codes\11\11.3\start-event\src\org\crazyit\activiti\HandleErrorDelegate.java),该类只会输出“8080端口关闭,开始处理...”一句话。代码清单11-12加载流程文件并启动流程。
代码清单11-12:codes\11\11.3\start-event\src\org\crazyit\activiti\ErrorStartEvent.java
//创建流程引擎
ProcessEngine engine = ProcessEngines.getDefaultProcessEngine();
//得到流程存储服务组件
RepositoryService repositoryService = engine.getRepositoryService();
//得到运行时服务组件
RuntimeService runtimeService = engine.getRuntimeService();
//部署流程文件
repositoryService.createDeployment()
.addClasspathResource("bpmn/ErrorStartEvent.bpmn").deploy();
//启动流程
runtimeService.startProcessInstanceByKey("errorStartProcess");
由于在本机并没有开启8080端口,因此运行代码清单11-12后,输出结果为:
开始检查8080端口
检查时出现异常,抛出错误
8080端口关闭,开始处理...
为了查看8080端口启动时的效果,需要将8080启动,启动方法可以使用一些web服务器或者直接使用编码方式启动,本例直接使用编启动方法,启动8080端口如代码清单11-13所示。
代码清单11-13:codes\11\11.3\start-event\src\org\crazyit\activiti\Server8080.java
ServerSocket serverSocket = new ServerSocket(8080);
while (true) {
}
直接使用ServerSocket建立8080端口,使用代码清单11-13开启了8080端口后,此时再次运行代码清单11-12,看到运行效果下:
开始检查8080端口
检查8080端口完成
当检查到8080端口已经开启,并没有抛出BpmnError,因此不会触发错误启动事件,也不会进入事件子流程。本小节中关于事件子流程、嵌套子流程和调用流程的内容,请见流程与子流程一章。
京东购买地址:https://item.jd.com/12246565.html
工作流Activiti6电子书:http://blog.csdn.net/boxiong86/article/details/78488562
工作流Activiti6教学视频:http://blog.csdn.net/boxiong86/article/details/78608585
本书代码共享地址:https://gitee.com/yangenxiong/CrazyActiviti