上一篇的示例中我们尝试用了jbpm Service API,现在我们仍然详细介绍API。
一, 流程引擎对象,ProcessEngine是jbpm4所有Service API之源
在jbpm中各种服务相互依存,但所有的service API都从ProcessEngine中获得,它是由
Configuration类构建的,即工作流引擎根据配置产生。
ProcessEngine是线程安全的,因此它可以保存在静态变量中,甚至JNDI命名服务
中或者其他重要位置。在应用中,所有线程和请求都可以使用同一个ProcessEngine对象,
以下代码告诉您如何获得ProcessEngine:
ProcessEngine processEngine=Configuration.getProcessEngine();
上面代码中的Configuration使用了classpath根目录下的默认配置文件jbpm.cfg.xml创
建一个ProcessEngine。如果你要自定义位置,那么可以这样做:
ProcessEngine processEngine=new Configuration().setResource(“myjbpm.xml”)
.buildProcessEngine();
然后,那6个service直接可以用processEngine.getXXX()得到。下面把这6个service描述一下:
RepositoryService—流程之源服务的借口。提供对流程定义的部署,查询,删除等操作。
ExecutionService—流程执行服务的借口。提供启动流程实例,“执行”推进,设置流程变量等操作
ManagementService—流程管理控制服务的接口,提供异步工作(Job)相关的执行和查询操作。
TaskService—人工任务服务的接口。提供对任务(Task)的创建,提交,查询,保存,删除等操作。
HistoryService—流程历史服务的接口。提供对流程历史库(即已完成的流程实例归
档)中历史流程实例,历史活动实例等记录的查询操作。还提供诸如某个流程定义中所有活动
的平均持续时间,某个流程定义中某转移的经过次数等数据分析服务。
IdentityService—身份认证服务的接口。提供对流程用户,用户组以及组成员关系的相关服务
二, 发起一个流程实例有这样几种方式
1, 常规方法:
ProcessInstance
processInstance=executionService.startProcessInstanceByKey(“order”);
2, 指定业务键发起流程实例,假如要对特定的业务实例来说,我们需要将每个流程
实例和一个独特的业务实例关联起来,比如一个保险流程的实例必然需要和一个保险单号关
联,一个订单流程的实例必然需要和一个订单号关联,以便业务上的查询或索引,那么我们就
可以通过为新启动的流程实例分配一个业务键来做到。一个业务键必须在此流程定义所有版本
的流程实例范围内是唯一的。比如以下:
ProcessInstance processInstance=executionService.startProcessInstanceByKey(“order”,”afei520”);
在上面种,afei520就是业务键,和特定的业务有关系。
注:最好指定一个业务键发起流程实例,一般情况下,在业务应用中,找到这样的业务键并不
困难,提供一个业务键的好处是可以根据业务来执行流程实例搜索,这是一个最佳实践。
3, 指定变量发起流程实例 如果新的流程实例需要一些输入参数启动,可以将这个参
数放在流程变量里,然后在发起流程时传入流程变量对象。代码如下:
Map<String, Object> valMap = new HashMap<String, Object>();
valMap.put("proposer", user);
//创建流程实例
return
executionService.startProcessInstanceByKey("loan",valMap);
三, 任务服务API,这里的任务是指jbpm task活动产生的人机交互业务。假如要展
示一个人的任务列表:
List<Task> tasks=taskService.findPersonalTasks(“afei”);
一般来说,任务会有一个表单,显示在一些用户界面中如jsp。这个表单需要读写与任务相关的流程数据,一般是通过任务变量的方式。如下代码:
String taskId=taskId;
Set<String> varis=taskService.getVariableNames()taskId;
//读取任务变量
Set<String> variableNames=taskService.getVariableNames(checkTask.getId());
for(Iterator<String> it=variableNames.iterator();it.hasNext();){
System.out.println("task vari:"+it.next());
}
Map<String,Object> variables=taskService.getVariables(checkTask.getId(),variableNames);
System.out.println("variables:"+variables);
或自行创建variables=new HashMap<String,Object>();
//设置”键值”形式的任务变量
Variables.put(“username”,”afei”);
Variables.put(“age”,111111);
//将变量存入任务
taskService.setVariables(taskId,variables);
四, 对于流程,任务,历史的查询是很重要的,从jbpm4开始,流程查询系统由一
组新的API来支持,这组API可以覆盖到大多数您能想到的查询。
比如流程实例查询:
List<ProcessInstance> results=executionService.
createProcessInstanceQuery().
processDefinitionId(“my_process_definition”).//流程定义Id
notSuspended().//未挂起
Page(0,50).list();//分页
比如任务的查询:
List<Task> tasks=taskService.createTaskQuery().
processInstanceId(piId).//流程实例
assignee(“Alex”).分配给Alex的任务
page(100,200).//分页
orderDesc(TaskQuery.PROPERTY_DUEDATE).
//根据日期逆向排序
List();
其他的服务以此类推
五, 下面看一个例子。
首先angel.jpdl.xml代码
<?xml version="1.0" encoding="UTF-8"?>
<process name="angel" xmlns="http://jbpm.org/4.4/jpdl">
<start name="start">
<transition name="to state" to="state"/>
</start>
<!-- state活动是等待活动。需要收到一个外部的执行信号才能流转通过。 -->
<state name="state">
<transition name="to task" to="task"/>
</state>
<!-- task活动是等待活动。在这里被分配给用户afei办理,afei办理完成提交任务后才能流转通过 -->
<task assignee="afei" name="task">
<transition name="to end" to="end"/>
</task>
<end name="end"/>
</process>
Java测试代码:
package org.test;
import org.jbpm.api.Execution;
import org.jbpm.api.ProcessInstance;
import org.jbpm.api.history.HistoryProcessInstance;
import org.jbpm.api.history.HistoryTask;
import org.jbpm.api.task.Task;
import org.jbpm.test.JbpmTestCase;
public class AngelTest extends JbpmTestCase {
String deploymentId;
@Override
public void setUp() throws Exception {
super.setUp();
deploymentId = repositoryService.createDeployment()
.addResourceFromClasspath("org/test/angel.jpdl.xml").deploy();
}
@Override
public void tearDown() throws Exception {
repositoryService.deleteDeploymentCascade(deploymentId);
super.tearDown();
}
public void testAfei() {
System.out.println("********testAfei start********");
// 使用执行服务:根据已部署流程定义的名称angel,发起流程实例
ProcessInstance pi = executionService
.startProcessInstanceByKey("angel");
// 获取流程实例ID
String pid = pi.getId();
// 获取当前活动的执行对象
Execution executionState = pi.findActiveExecutionIn("state");
// 断言当前活动即为state
assertNotNull(executionState);
// 使用执行服务:发出执行信号结束当前活动,继续流程的执行
executionService.signalExecutionById(executionState.getId());
// 使用执行服务:从持久化层中获取“最新”的流程实例对象
pi = executionService.findProcessInstanceById(pid);
Execution executionTask = pi.findActiveExecutionIn("task");
// 断言当前活动即为task
assertNotNull(executionTask);
// 使用任务服务:获取用户afei的任务,即task活动产生的任务
Task task = taskService.findPersonalTasks("afei").get(0);
// 使用任务服务:完成任务
taskService.completeTask(task.getId());
// 使用历史服务:创建历史任务查询
HistoryTask historyTask = historyService.createHistoryTaskQuery()
.taskId(task.getId()).uniqueResult();
// 断言上一步完成的任务已成为历史,即可通过历史任务查询获取之
assertNotNull(historyTask);
// 断言该流程实例已经结束
assertProcessInstanceEnded(pid);
// 使用历史服务:创建历史流程实例查询
HistoryProcessInstance hpi = historyService
.createHistoryProcessInstanceQuery().processInstanceId(pid)
.uniqueResult();
//断言该流程实例已经成为历史,即可通过历史流程实例查询获取它
assertNotNull(hpi);
//单元测试至此执行完毕
System.out.println("********testAfei end********");
}
}
这个例子主要是进一步看看Service API的功能及其使用场景。这个例子基本上走完了一个流程的完整生命周期,下一篇着重介绍Jbpm的核心————jPDL。