相关代码参考gihub:https://github.com/changwensir/Activiti/tree/master/ActivitiDemo1
前面的话,我们使用的是classpath加载资源文件方式来部署流程定义的,但是这种方式有局限性,只能适合小项目,固定写死的流程;
实际项目的话,需要来动态导入流程定义文件,通过把bpmn和png文件打包成zip压缩包,然后用户界面直接导入到系统,然后解析,部署流程定义;Activiti是支持这种方式的。今天我们来实现下这种方式;package com.java1234.activiti.test; import org.activiti.engine.ProcessEngine; import org.activiti.engine.ProcessEngines; import org.activiti.engine.repository.Deployment; import org.junit.Test; public class ProcessDefinition { /** * 获取默认的流程引擎实例 会自动读取activiti.cfg.xml文件 */ private ProcessEngine processEngine=ProcessEngines.getDefaultProcessEngine(); /** * 部署流程定义使用classpath方式 */ @Test public void deployWithClassPath(){ // 获取部署对象 Deployment deployment=processEngine.getRepositoryService() // 部署Service .createDeployment() // 创建部署 .addClasspathResource("diagrams/helloWorld.bpmn") // 加载资源文件 .addClasspathResource("diagrams/helloWorld.png") // 加载资源文件 .name("HelloWorld流程") // 流程名称 .deploy(); // 部署 System.out.println("流程部署ID:"+deployment.getId()); System.out.println("流程部署Name:"+deployment.getName()); } }
这一步在数据库中将操作三张表:
a).act_re_deployment(部署对象表)
存放流程定义的显示名和部署时间,每部署一次增加一条记录
b).act_re_procdef(流程定义表)
存放流程定义的属性信息,部署每个新的流程定义都会在这张表中增加一条记录。
注意:当流程定义的key相同的情况下,使用的是版本升级
c).act_ge_bytearray(资源文件表)
存储流程定义相关的部署信息。即流程定义文档的存放地。每部署一次就会增加两条记录,一条是关于bpmn规则文件的,一条是图片的(如果部署时只指定了bpmn一个文件,activiti会在部署时解析bpmn文件内容自动生成流程图)。两个文件不是很大,都是以二进制形式存储在数据库中。
下面我们用zip方式来实现,新建一个deployWithZip方法:
/** * 部署流程定义使用zip方式 */ @Test public void deployWithZip(){ InputStream inputStream=this.getClass() // 获取当前class对象 .getClassLoader() // 获取类加载器 .getResourceAsStream("diagrams/helloWorld.zip"); // 获取指定文件资源流 ZipInputStream zipInputStream=new ZipInputStream(inputStream); // 实例化zip输入流对象 // 获取部署对象 Deployment deployment=processEngine.getRepositoryService() // 部署Service .createDeployment() // 创建部署 .name("HelloWorld流程2") // 流程名称 .addZipInputStream(zipInputStream) // 添加zip是输入流 .deploy(); // 部署 System.out.println("流程部署ID:"+deployment.getId()); System.out.println("流程部署Name:"+deployment.getName()); }我们运行这个测试类:
SLF4J: Failed to load class "org.slf4j.impl.StaticLoggerBinder".
SLF4J: Defaulting to no-operation (NOP) logger implementation
SLF4J: See http://www.slf4j.org/codes.html#StaticLoggerBinder for further details.
流程部署ID:7501
流程部署Name:HelloWorld流程2
|
我们会发现,这里多了一条记录;
act_re_procdef 流程定义表
流程定义表里,我们发现VERSION_字段 版本升级了,KEY依然是一样的;
act_ge_bytearry 资源文件表
资源表里相应的多里两条记录;
这里再提一个表 act_ge_property 属性表
这里的next_dbid是主键策略,就是规定好了下一次生成的id就是10001;
/**查询流程定义*/ @Test public void findProcessDefinition(){ List<ProcessDefinition> list = processEngine.getRepositoryService()//与流程定义和部署对象相关的Service .createProcessDefinitionQuery()//创建一个流程定义的查询 /**指定查询条件,where条件*/ // .deploymentId(deploymentId)//使用部署对象ID查询 // .processDefinitionId(processDefinitionId)//使用流程定义ID查询 // .processDefinitionKey(processDefinitionKey)//使用流程定义的key查询 // .processDefinitionNameLike(processDefinitionNameLike)//使用流程定义的名称模糊查询 /**排序*/ .orderByProcessDefinitionVersion().asc()//按照版本的升序排列 // .orderByProcessDefinitionName().desc()//按照流程定义的名称降序排列 /**返回的结果集*/ .list();//返回一个集合列表,封装流程定义 // .singleResult();//返回惟一结果集 // .count();//返回结果集数量 // .listPage(firstResult, maxResults);//分页查询 if(list!=null && list.size()>0){ for(ProcessDefinition pd:list){ System.out.println("流程定义ID:"+pd.getId());//流程定义的key+版本+随机生成数 System.out.println("流程定义的名称:"+pd.getName());//对应helloworld.bpmn文件中的name属性值 System.out.println("流程定义的key:"+pd.getKey());//对应helloworld.bpmn文件中的id属性值 System.out.println("流程定义的版本:"+pd.getVersion());//当流程定义的key值相同的相同下,版本升级,默认1 System.out.println("资源名称bpmn文件:"+pd.getResourceName()); System.out.println("资源名称png文件:"+pd.getDiagramResourceName()); System.out.println("部署对象ID:"+pd.getDeploymentId()); System.out.println("#########################################################"); } } }
流程定义ID:helloworld:1:4 流程定义的名称:helloworldProcess 流程定义的key:helloworld 流程定义的版本:1 资源名称bpmn文件:diagrams/helloWorld.bpmn 资源名称png文件:diagrams/helloWorld.png 部署对象ID:1 ######################################################### 流程定义ID:helloworld:2:304 流程定义的名称:helloworldProcess 流程定义的key:helloworld 流程定义的版本:2 资源名称bpmn文件:diagrams/helloWorld.bpmn 资源名称png文件:diagrams/helloWorld.png 部署对象ID:301 ######################################################### 流程定义ID:helloworld:3:404 流程定义的名称:helloworldProcess 流程定义的key:helloworld 流程定义的版本:3 资源名称bpmn文件:helloworld.bpmn 资源名称png文件:helloworld.png 部署对象ID:401 ######################################################### |
/***附加功能:查询最新版本的流程定义*/ @Test public void findLastVersionProcessDefinition(){ List<ProcessDefinition> list = processEngine.getRepositoryService()// .createProcessDefinitionQuery()// .orderByProcessDefinitionVersion().asc()//使用流程定义的版本升序排列(可能存在多个集合) .list(); /** * Map<String,ProcessDefinition> * map集合的key:流程定义的key * map集合的value:流程定义的对象 * map集合的特点:当map集合key值相同的情况下,后一次的值将替换前一次的值 */ Map<String, ProcessDefinition> map = new LinkedHashMap<String, ProcessDefinition>(); if(list!=null && list.size()>0){ for(ProcessDefinition pd:list){ map.put(pd.getKey(), pd); } } List<ProcessDefinition> pdList = new ArrayList<ProcessDefinition>(map.values()); if(pdList!=null && pdList.size()>0){ for(ProcessDefinition pd:pdList){ System.out.println("流程定义ID:"+pd.getId());//流程定义的key+版本+随机生成数 System.out.println("流程定义的名称:"+pd.getName());//对应helloworld.bpmn文件中的name属性值 System.out.println("流程定义的key:"+pd.getKey());//对应helloworld.bpmn文件中的id属性值 System.out.println("流程定义的版本:"+pd.getVersion());//当流程定义的key值相同的相同下,版本升级,默认1 System.out.println("资源名称bpmn文件:"+pd.getResourceName()); System.out.println("资源名称png文件:"+pd.getDiagramResourceName()); System.out.println("部署对象ID:"+pd.getDeploymentId()); System.out.println("#########################################################"); } } }
/**删除流程定义*/ @Test public void deleteProcessDefinition(){ //使用部署ID,完成删除 String deploymentId = "601"; /** * 不带级联的删除 * 只能删除没有启动的流程,如果流程启动,就会抛出异常 */ // processEngine.getRepositoryService()// // .deleteDeployment(deploymentId); /** * 级联删除 * 不管流程是否启动,都能可以删除 */ processEngine.getRepositoryService()// .deleteDeployment(deploymentId, true); System.out.println("删除成功!"); }说明:
/**附加功能:删除流程定义(删除key相同的所有不同版本的流程定义)*/ @Test public void deleteProcessDefinitionByKey(){ //流程定义的key String processDefinitionKey = "helloworld"; //先使用流程定义的key查询流程定义,查询出所有的版本 List<ProcessDefinition> list = processEngine.getRepositoryService()// .createProcessDefinitionQuery()// .processDefinitionKey(processDefinitionKey)//使用流程定义的key查询 .list(); //遍历,获取每个流程定义的部署ID if(list!=null && list.size()>0){ for(ProcessDefinition pd:list){ //获取部署ID String deploymentId = pd.getDeploymentId(); processEngine.getRepositoryService()// .deleteDeployment(deploymentId, true); } } }
/** * 通过流程部署ID获取流程图图片 */ @Test public void getImageById()throws Exception{ InputStream inputStream=processEngine.getRepositoryService() .getResourceAsStream("10001", "helloWorld.png"); // 根据流程部署ID和资源名称获取输入流 FileUtils.copyInputStreamToFile(inputStream, new File("D:/helloWorld.png")); } /**查看流程图 * @throws IOException */ @Test public void viewPic() throws IOException{ /**将生成图片放到文件夹下*/ //部署Id String deploymentId = "301"; //获取图片资源名称 List<String> list = processEngine.getRepositoryService()// .getDeploymentResourceNames(deploymentId); //定义图片资源的名称 String resourceName = ""; if(list!=null && list.size()>0){ for(String name:list){ if(name.indexOf(".png")>=0){ resourceName = name; } } } //获取图片的输入流 InputStream in = processEngine.getRepositoryService()// .getResourceAsStream(deploymentId, resourceName); //将图片生成到D盘的目录下 File file = new File("D:/"+resourceName); //将输入流的图片写到D盘下 FileUtils.copyInputStreamToFile(in, file); }说明:
首先说下结论,流程定义是不能修改的;这里举例子,假如一个流程定义的流程实例在活动运行中。假如可以修改,本来要流转到A这个节点,因为流程定义修改了,流转到B这个节点。就不符合当时这个流程实例的初衷了;所以一般开发的话,不能修改流程定义,我们是通过增加版本号的方式。来实现“修改”的;
总结:
Deployment 部署对象
1、一次部署的多个文件的信息。对于不需要的流程可以删除和修改。
2、对应的表:
act_re_deployment:部署对象表
act_re_procdef:流程定义表
act_ge_bytearray:资源文件表
act_ge_property:主键生成策略表
ProcessDefinition 流程定义
1、解析.bpmn后得到的流程定义规则的信息,工作流系统就是按照流程定义的规则执行的。