1.初始化数据库

使用工作流引擎创建23张表

public class TestActiviti {
/**
* 使用代码创建工作流需要的23张表
*/
@Test
public void createTable() {
ProcessEngineConfiguration createStandaloneProcessEngineConfiguration = ProcessEngineConfiguration.createStandaloneProcessEngineConfiguration();
createStandaloneProcessEngineConfiguration.setJdbcDriver("com.mysql.jdbc.Driver");
createStandaloneProcessEngineConfiguration.setJdbcUrl("jdbc:mysql://localhost:3306/activiti0710?useUnicode=true&characterEncoding=utf8");
createStandaloneProcessEngineConfiguration.setJdbcUsername("root"); /**
public static final String DB_SCHEMA_UPDATE_FALSE = "false"; 不能自动创建表,需要表存在
public static final String DB_SCHEMA_UPDATE_CREATE_DROP = "create-drop"; 表示先删除表,在创建表
public static final String DB_SCHEMA_UPDATE_TRUE = "true";如果表不存在,自动创建表
*/
createStandaloneProcessEngineConfiguration.setDatabaseSchemaUpdate(createStandaloneProcessEngineConfiguration.DB_SCHEMA_UPDATE_TRUE);
//工作流的核心对象
ProcessEngine buildProcessEngine = createStandaloneProcessEngineConfiguration.buildProcessEngine();
System.out.println(buildProcessEngine); }
}

在Activiti中,在创建核心的流程引擎对象时会自动建表。如果程序正常执行,mysql会自动建库,然后创建23张表。

2.添加并制定配置文件

在Actiiti5中定制流程必定会操作到数据库,如果都像上面那样写一大段代码会非常麻烦,所以我们可以把数据库连接配置写入配置文件。

创建activiti.cfg.xml文件,配置

<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:context="http://www.springframework.org/schema/context" xmlns:tx="http://www.springframework.org/schema/tx"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd
http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.0.xsd"> <!--
ProcessEngineConfiguration createStandaloneProcessEngineConfiguration = ProcessEngineConfiguration.createStandaloneProcessEngineConfiguration();
createStandaloneProcessEngineConfiguration.setJdbcDriver("com.mysql.jdbc.Driver");
createStandaloneProcessEngineConfiguration.setJdbcUrl("jdbc:mysql://localhost:3306/activiti0710?useUnicode=true&characterEncoding=utf8");
createStandaloneProcessEngineConfiguration.setJdbcUsername("root"); /**
public static final String DB_SCHEMA_UPDATE_FALSE = "false"; 不能自动创建表,需要表存在
public static final String DB_SCHEMA_UPDATE_CREATE_DROP = "create-drop"; 表示先删除表,在创建表
public static final String DB_SCHEMA_UPDATE_TRUE = "true";如果表不存在,自动创建表
*/
createStandaloneProcessEngineConfiguration.setDatabaseSchemaUpdate(createStandaloneProcessEngineConfiguration.DB_SCHEMA_UPDATE_TRUE);
//工作流的核心对象
ProcessEngine buildProcessEngine = createStandaloneProcessEngineConfiguration.buildProcessEngine(); -->
<bean id="processEngineConfiguration" class="org.activiti.engine.impl.cfg.StandaloneInMemProcessEngineConfiguration">
<!-- 连接数据库配置 -->
<property name="jdbcDriver" value="com.mysql.jdbc.Driver"></property>
<property name="jdbcUrl" value="jdbc:mysql://localhost:3306/activiti0710?useUnicode=true&amp;characterEncoding=utf8"></property>
<property name="jdbcUsername" value="root"></property>
<!-- 没有表创建表 -->
<property name="databaseSchemaUpdate" value="true"></property>
</bean> </beans>

java代码:

/**
* 使用配置文件创建工作流需要的23张表
*/
@Test
public void createTable_2() {
ProcessEngineConfiguration createProcessEngineConfigurationFromResource = ProcessEngineConfiguration.createProcessEngineConfigurationFromResource("activiti.cfg.xml");
ProcessEngine buildProcessEngine = createProcessEngineConfigurationFromResource.buildProcessEngine();
System.out.println(buildProcessEngine);
}

 3. HelloWorld程序(模拟流程的执行)

先新建一个bpmn文件

编写junit测试类

public class helloWorld {

    ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();
/**
* 部署流程定义
*/
@Test
public void deployProcessDefinition() {
Deployment deploy = processEngine.getRepositoryService()//与流程定义和部署对象相关的Service
.createDeployment()//创建一个部署对象
.name("helloWorld入门程序")//添加部署的名称
.addClasspathResource("diagrams/HelloWorld.bpmn")//一次只能加载一个文件
.addClasspathResource("diagrams/HelloWorld.png")
.deploy();//完成部署
System.out.println(deploy.getId());
System.out.println(deploy.getName());
} /**
* 启动流程实例
*/
@Test
public void startProcessInstance() {
//流程定义的key
String processDefinitionKey = "helloWorld";
ProcessInstance startProcessInstanceByKey = processEngine.getRuntimeService()//与正在执行的流程实例和执行对象相关的Service
.startProcessInstanceByKey(processDefinitionKey);//使用流程定义的key启动流程实例,key对应HelloWorld.bpmn中的id属性,使用key启动,默认是最新版本的流程定义启动
System.out.println("流程实例id" + startProcessInstanceByKey.getId());//流程实例id
System.out.println("流程定义id" + startProcessInstanceByKey.getProcessDefinitionId());//流程定义id
} /**
* 查询当前人的个人任务
*/
@Test
public void findMyPersonTask() {
String assignee = "王五";
List<Task> list = processEngine.getTaskService()//与正在执行的任务管理相关的service
.createTaskQuery()//创建任务查询对象
.taskAssignee(assignee)//指定个人任务查询,指定办理人
.list();
if (list != null && list.size() > 0) {
for (Task task : list) {
System.out.println("任务ID:"+task.getId());
System.out.println("任务名称:"+task.getName());
System.out.println("任务创建时间:"+task.getCreateTime());
System.out.println("任务办理人:"+task.getAssignee());
System.out.println("流程实例ID:"+task.getProcessInstanceId());
System.out.println("执行对象ID:"+task.getExecutionId());
System.out.println("流程定义ID:"+task.getProcessDefinitionId());
}
}
} /**
* 完成个人任务
*/
@Test
public void completMyPersonTask() {
String taskId = "302";
processEngine.getTaskService()//与正在执行的任务管理相关的service
.complete(taskId);
System.out.println("完成任务:任务ID"+taskId);
}
}

4.部署流程定义(classpath路径加载文件)

说明:

1) 先获取流程引擎对象:在创建时会自动加载classpath下的activiti.cfg.xml

2) 首先获得默认的流程引擎,通过流程引擎获取了一个RepositoryService对象(仓库对象)

3) 由仓库的服务对象产生一个部署对象配置对象,用来封装部署操作的相关配置。

4) 这是一个链式编程,在部署配置对象中设置显示名,上传流程定义规则文件

5) 向数据库表中存放流程定义的规则信息。

6) 这一步在数据库中将操作三张表:

a) act_re_deployment(部署对象表)

存放流程定义的显示名和部署时间,每部署一次增加一条记录

b) act_re_procdef(流程定义表)

存放流程定义的属性信息,部署每个新的流程定义都会在这张表中增加一条记录。

注意:当流程定义的key相同的情况下,使用的是版本升级

c) act_ge_bytearray(资源文件表)

存储流程定义相关的部署信息。即流程定义文档的存放地。每部署一次就会增加两条记录,一条是关于bpmn规则文件的,一条是图片的(如果部署时只指定了bpmn一个文件,activiti会在部署时解析bpmn文件内容自动生成流程图)。两个文件不是很大,都是以二进制形式存储在数据库中。

/**
* 部署流程定义(从classpath)
*/
@Test
public void deployProcessDefinition_classpath() {
Deployment deploy = processEngine.getRepositoryService()//与流程定义和部署对象相关的Service
.createDeployment()//创建一个部署对象
.name("流程定义")//添加部署的名称
.addClasspathResource("diagrams/HelloWorld.bpmn")//一次只能加载一个文件
.addClasspathResource("diagrams/HelloWorld.png")
.deploy();//完成部署
System.out.println(deploy.getId());
System.out.println(deploy.getName());
}

5. 部署流程定义(zip格式文件)

/**
* 部署流程定义(从zip)
*/
@Test
public void deployProcessDefinition_zip() {
InputStream in = this.getClass().getClassLoader().getResourceAsStream("diagrams/HelloWorld.zip");
ZipInputStream zipInputStream = new ZipInputStream(in);
Deployment deploy = processEngine.getRepositoryService()//与流程定义和部署对象相关的Service
.createDeployment()//创建一个部署对象
.name("流程定义")//添加部署的名称
.addZipInputStream(zipInputStream)//一次只能加载一个文件
.deploy();//完成部署
System.out.println(deploy.getId());
System.out.println(deploy.getName());
}

6. 查看流程定义

查询流程定义的信息

/**
* 查询流程定义
*/
@Test
public void findProcessDefinition() {
List<ProcessDefinition> list = processEngine.getRepositoryService()//与流程定义和部署对象相关的Service
.createProcessDefinitionQuery()//创建流程定义查询
// .deploymentId(deploymentId) //使用部署对象ID查询
// .processDefinitionKey(processDefinitionKey) //使用流程定义的key查询
// .processDefinitionId(processDefinitionId) //使用流程定义的id查询
// .processDefinitionKeyLike(processDefinitionKeyLike) //使用流程定义的id模糊查询
// .processDefinitionNameLike(processDefinitionNameLike) //使用流程定义的name模糊查询
/** 排序 **/
.orderByProcessDefinitionVersion().asc() //按照版本升序排列
// .orderByProcessDefinitionName().desc() // 按照流程定义的名称降序排序
.list(); //返回一个集合列表,封装流程定义
// .singleResult() //返回唯一结果集
// .listPage(firstResult, maxResults) //分页查询
// .count() //返回结果集数量
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());
}
} }

说明:

1) 流程定义和部署对象相关的Service都是RepositoryService。

2) 创建流程定义查询对象,可以在ProcessDefinitionQuery上设置查询的相关参数

3) 调用ProcessDefinitionQuery对象的list方法,执行查询,获得符合条件的流程定义列表

4) 由运行结果可以看出:

Key和Name的值为:bpmn文件process节点的id和name的属性值

5) key属性被用来区别不同的流程定义。

6) 带有特定key的流程定义第一次部署时,version为1。之后每次部署都会在当前最高版本号上加1

7) Id的值的生成规则为:{processDefinitionKey}:{processDefinitionVersion}:{generated-id}, 这里的generated-id是一个自动生成的唯一的数字

8) 重复部署一次,deploymentId的值以一定的形式变化

规则act_ge_property表生成

 7.删除流程定义

删除部署到activiti中的流程定义。

    /**
* 删除流程定义
*/
@Test
public void delProcessDefinition() {
//使用部署ID,完成删除
String deploymentId = "601";
/**
* 不带级联删除
* 只能删除没有启动的流程,如果流程启动,就会抛出异常
*/
/*processEngine.getRepositoryService()
.deleteDeployment(deploymentId);*/
/**
* 级联删除
* 不管流程是否启动,都能删除
*/
processEngine.getRepositoryService()
.deleteDeployment(deploymentId, true);
System.out.println("删除成功!");
}

8. 获取流程定义文档的资源(查看流程图附件)

查询出流程定义文档。主要查的是图片,用于显示流程用。

/**
* 查看流程图
* @throws IOException
*/
@Test
public void viewPic() throws IOException {
//将生成的图片放置到文件夹下
String deploymentId = "901";
//获取图片资源名称
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);
FileUtils.copyInputStreamToFile(in, file);
}

说明:

1) deploymentId为流程部署ID

2) resourceName为act_ge_bytearray表中NAME_列的值

3) 使用repositoryService的getDeploymentResourceNames方法可以获取指定部署下得所有文件的名称

4) 使用repositoryService的getResourceAsStream方法传入部署ID和资源图片名称可以获取部署下指定名称文件的输入流

最后的有关IO流的操作,使用FileUtils工具的copyInputStreamToFile方法完成流程流程到文件的拷贝,将资源文件以流的形式输出到指定文件夹下

9.流程定义是无法修改的

控制修改:使用流程定义的key相同情况下,版本升级

10.附加功能:查询最新版本的流程定义

/**
* 附加功能:查询最新版本的流程定义
*/
@Test
public void findLastVersionProcessDefinition() {
List<ProcessDefinition> list = processEngine.getRepositoryService()
.createProcessDefinitionQuery()
.orderByProcessDefinitionVersion().desc()//使用流程定义的版本升序排列
.list(); /**
* map集合的key:流程定义的key
* map集合的value:流程定义的对象
* map集合的特点:当map集合key相同的情况下,后一次的值将替换前一次的值
*/
Map<String, ProcessDefinition> map = new LinkedHashMap<>();
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());
}
}
}

11.附加功能:删除流程定义(删除key相同的所有不同版本的流程定义)

/**
* 附加功能: 删除流程定义(删除key相同的所有流程版本不同的流程定义)
*/
@Test
public void delProcessDefinitionByKey() {
//流程定义的key
String processDefinitionKey = "helloWorld";
//先使用流程定义的key查询流程定义,查询所有的版本
List<ProcessDefinition> list = processEngine.getRepositoryService()
.createProcessDefinitionQuery()
.processDefinitionKey(processDefinitionKey)
.list();
if (list != null && list.size() > 0) {
for (ProcessDefinition pd : list) {
String deploymentId = pd.getDeploymentId();
processEngine.getRepositoryService().deleteDeployment(deploymentId, true);
}
}
}

12.流程实例,任务和执行

Execution   执行对象

按流程定义的规则执行一次的过程.

对应的表:

act_ru_execution: 正在执行的信息

act_hi_procinst:已经执行完的历史流程实例信息

act_hi_actinst:存放历史所有完成的活动

ProcessInstance  流程实例

特指流程从开始到结束的那个最大的执行分支,一个执行的流程中,流程实例只有1个。

注意

(1)如果是单例流程,执行对象ID就是流程实例ID

(2)如果一个流程有分支和聚合,那么执行对象ID和流程实例ID就不相同

(3)一个流程中,流程实例只有1个,执行对象可以存在多个。

Task 任务

执行到某任务环节时生成的任务信息。

对应的表:

act_ru_task:正在执行的任务信息

act_hi_taskinst:已经执行完的历史任务信息

Activiti学习记录(二)的更多相关文章

  1. Material Calendar View 学习记录(二)

    Material Calendar View 学习记录(二) github link: material-calendarview; 在学习记录一中简单翻译了该开源项目的README.md文档.接下来 ...

  2. Spring Boot学习记录(二)--thymeleaf模板 - CSDN博客

    ==他的博客应该不错,没有细看 Spring Boot学习记录(二)--thymeleaf模板 - CSDN博客 http://blog.csdn.net/u012706811/article/det ...

  3. JavaScript学习记录二

    title: JavaScript学习记录二 toc: true date: 2018-09-13 10:14:53 --<JavaScript高级程序设计(第2版)>学习笔记 要多查阅M ...

  4. 2.VUE前端框架学习记录二

    VUE前端框架学习记录二:Vue核心基础2(完结)文字信息没办法描述清楚,主要看编码实战里面,有附带有一个完整可用的Html页面,有需要的同学到脑图里面自取.脑图地址http://naotu.baid ...

  5. 【Activiti学习之二】Activiti API(一)

    环境 JDK 1.8 MySQL 5.6 Tomcat 7 Eclipse-Luna activiti 6.0 一.Activiti数据查询准备数据: package com.wjy.act; imp ...

  6. Activiti学习(二)数据表结构

    Activiti工作流引擎数据库表结构 数据库表的命名 Acitiviti数据库中表的命名都是以ACT_开头的.第二部分是一个两个字符用例表的标识.此用例大体与服务API是匹配的. l        ...

  7. Spring Boot学习记录(二)–thymeleaf模板

    自从来公司后都没用过jsp当界面渲染了,因为前后端分离不是很好,反而模板引擎用的比较多,thymeleaf最大的优势后缀为html,就是只需要浏览器就可以展现页面了,还有就是thymeleaf可以很好 ...

  8. 【java并发编程艺术学习】(四)第二章 java并发机制的底层实现原理 学习记录(二) synchronized

    章节介绍 本章节主要学习 Java SE 1.6 中为了减少获得锁 和 释放锁 时带来的性能消耗 而引入的偏向锁 和 轻量级锁,以及锁的存储结构 和 升级过程. synchronized实现同步的基础 ...

  9. SpringBoot学习记录(二)

    一. SpringBoot日志框架 SpringBoot:底层是Spring框架,Spring框架默认是用JCL(commons-logging): SpringBoot选用SLF4j和logback ...

随机推荐

  1. 点分治【bzoj1468】 Tree

    点分治[bzoj1468] Tree Description 给你一棵TREE,以及这棵树上边的距离.问有多少对点它们两者间的距离小于等于K Input N(n<=40000) 接下来n-1行边 ...

  2. Python网络编程(一)

    最近在啃<python核心编程(第三版)>,感觉这本书并不是特别的友好,虽然有基于python3提出的改进代码:但是整书的基准感觉还是在python2.7.所以python3的代码中还是有 ...

  3. 6、python数据类型之元组(dict)

    字典字典的每个元素就是一个键值对,格式如下key:value{key1:value1,key2:value2,......} 1.创建 dict_eg = { "name":&qu ...

  4. Ubuntu14.04升级到Ubuntu16.04

    Ubuntu14.04升级到Ubuntu16.04 1.查看目前版本 lsb_release -a 2.执行更新命令 apt-get update && apt-get dist-up ...

  5. spring动态线程池(实质还是用了java的线程池)

    <?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.sp ...

  6. 【ACM】最长公共子序列 - 动态规划

    最长公共子序列 时间限制:3000 ms  |  内存限制:65535 KB 难度:3   描述 咱们就不拐弯抹角了,如题,需要你做的就是写一个程序,得出最长公共子序列.tip:最长公共子序列也称作最 ...

  7. linux目录权限

    linux中,有三种不同类型的用户可以对文件或目录进行访问:文件所有者,同组用户,其他用户.所有者一般是文件的创建者,文件所有者自动拥有对该文件的读.写和可执行权限.所有者能允许同组用户有权访问文件, ...

  8. windows 2012 r2 x64 安装IIS注意事项

    详细安装可以参考下面; https://jingyan.baidu.com/article/93f9803f234eade0e46f559f.html 下面只说一些注意事项,如果项目要用到wcf 的话 ...

  9. 19.CentOS7下PostgreSQL安装过程

    CentOS7下PostgreSQL安装过程 装包 sudo yum install postgresql-server postgresql-contrib 说明: 这种方式直接明了,其他方法也可以 ...

  10. Arduino连接pH计

    关于arduino连接ph,核心的东西就是对ph传感器返回的信号值进行一系列的操作,注意因为返回的信号很弱,而且外部环境对其影响也很大,所以需要在电路设计上加入一些功能,比如信号放大.滤波等,电路设计 ...