工作流框架--Activiti6.0(二)
-
----编写一个Activiti的程序
我们大概要完成4步。
(1)设计流程
(2)配置流程引擎
(3)加载流程文件
(4)启动流程
一、新建一个springboot项目 (版本:1.5.10和2.1.2版本同时测试)
1.1 新建springboot项目的版本为1.5.10
--activiti6.0不支持springboot的2.x 版本。 根据官方文档解释:https://www.baeldung.com/spring-activiti。
1.2 在src/main/resources下面新建一个prosesses文件夹
如果不建立,项目会报错。
二、设计流程
2.1 使用Eclipse-Activiti插件生成一张流程图。
起名为:first.bpmn(建立流程图)
2.2 可以在该图片上右键-->打开方式-->xml编辑器来查看文件内容(上面的项目是2.1.2版本 下面的项目是1.5.10版本,内容相同)
2.3 生成28张表
在properties中添加一些属性,各个Activiti的属性配置含义如下:
# 是否更新数据库表
spring.activiti.databaseSchemaUpdate=true
# 是否激活异步执行器
spring.activiti.asyncExecutorActivate=false
# 流程历史记录登录
spring.activiti.historyLevel=audit
# 是否检查更新流程定义
spring.activiti.checkProcessDefinitions=false
# 流程定义所在前缀
spring.activiti.processDefinitionLocationPrefix=classpath*:/procDef/
# 流程定义后缀
spring.activiti.processDefinitionLocationSuffixes=**.bpmn
# 部署流程定义时是否生成图片
spring.activiti.createDiagramOnDeploy=false
# 字体 下面内容为转成unicode的'宋体'
spring.activiti.activityFontName=\u5b8b\u4f53
spring.activiti.labelFontName=\u5b8b\u4f53
(1)application.properties
server.port: 8081
spring.datasource.url=jdbc:mysql://localhost:3306/activiti?useSSL=true&characterEncoding=utf8
spring.datasource.driverClassName=com.mysql.jdbc.Driver
spring.datasource.username=root
spring.datasource.password=root
activiti.databaseSchemaUpdate=true
activiti.checkProcessDefinitions=true
(2)新建一个测试类,创建28张表(在这行代码运行的时候,流程定义的部署已经执行..不过执行了一个默认的数据)
package com.example.demo;
import org.activiti.engine.ProcessEngines;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;
@RunWith(SpringRunner.class)
@SpringBootTest
public class ActivitiSpringboot1ApplicationTests {
@Test
public void contextLoads() {
ProcessEngines.getDefaultProcessEngine();
}
}
注意:在boot2.0 的时候,系统可能会报一个异常。原因:mysql和本机时差问题
Caused by: com.mysql.cj.exceptions.InvalidConnectionAttributeException:
The server time zone value 'Öйú±ê׼ʱ¼ä' is unrecognized or represents more than one time zone.
You must configure either the server or JDBC driver (via the serverTimezone configuration property) to use a more specifc time zone value if you want to utilize time zone support.
解决办法:
打开mysql cmd 的命令行,输入密码进入后,直接输入一行命令:
set global time_zone='+8:00'
(3)运行测试类
发现boot1.5.10 版本 和boot 2.1.2 均可以生成这28张表。
但是boot2.1.2 此时又报了异常,不能创建bean
org.springframework.beans.factory.BeanCreationException:
Error creating bean with name 'org.activiti.spring.boot.SecurityAutoConfiguration': Initialization of bean failed;
nested exception is java.lang.ArrayStoreException: sun.reflect.annotation.TypeNotPresentExceptionProxy
解决办法:
springboot Test 测试类中如何排除一个bean类
2.4 流程定义的部署
(1)配置类代码
package com.example;
import java.io.IOException;
import javax.sql.DataSource;
import org.activiti.engine.HistoryService;
import org.activiti.engine.IdentityService;
import org.activiti.engine.ManagementService;
import org.activiti.engine.ProcessEngine;
import org.activiti.engine.RepositoryService;
import org.activiti.engine.RuntimeService;
import org.activiti.engine.TaskService;
import org.activiti.spring.SpringProcessEngineConfiguration;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.io.Resource;
import org.springframework.core.io.ResourceLoader;
import org.springframework.core.io.support.PathMatchingResourcePatternResolver;
import org.springframework.jdbc.datasource.DataSourceTransactionManager;
import org.springframework.transaction.PlatformTransactionManager;
@Configuration
public class ActivitiConfiguration {
@Autowired
private DataSource dataSource;
@Autowired
private PlatformTransactionManager platformTransactionManager;
@Bean
public ProcessEngine processEngine(DataSourceTransactionManager transactionManager, DataSource dataSource) throws IOException {
SpringProcessEngineConfiguration configuration = new SpringProcessEngineConfiguration();
//自动部署已有的流程文件
Resource[] resources = new PathMatchingResourcePatternResolver().getResources(ResourceLoader.CLASSPATH_URL_PREFIX + "processes/*.bpmn");
configuration.setTransactionManager(transactionManager);
configuration.setDataSource(dataSource);
configuration.setDatabaseSchemaUpdate("true");
configuration.setDeploymentResources(resources);
configuration.setDbIdentityUsed(false);
return configuration.buildProcessEngine();
}
@Bean
public RepositoryService repositoryService(ProcessEngine processEngine) {
return processEngine.getRepositoryService();
}
@Bean
public RuntimeService runtimeService(ProcessEngine processEngine) {
return processEngine.getRuntimeService();
}
@Bean
public TaskService taskService(ProcessEngine processEngine) {
return processEngine.getTaskService();
}
@Bean
public HistoryService historyService(ProcessEngine processEngine) {
return processEngine.getHistoryService();
}
@Bean
public ManagementService managementService(ProcessEngine processEngine) {
return processEngine.getManagementService();
}
@Bean
public IdentityService identityService(ProcessEngine processEngine) {
return processEngine.getIdentityService();
}
}
(2)部署类代码
package com.example.demo;
import org.activiti.engine.RepositoryService;
import org.activiti.engine.repository.Deployment;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;
@RunWith(SpringRunner.class)
@SpringBootTest
public class ActivitiSpringboot1ApplicationTests {
private static final Logger log = LoggerFactory.getLogger(ActivitiSpringboot1ApplicationTests.class);
@Autowired
private RepositoryService repositoryService;
/**
* 流程定义的部署
*/
@Test
public void contextLoads() {
Deployment deployment = repositoryService // 与流程定义和部署对象相关的Service
.createDeployment()// 创建一个部署对象
.name("请假流程") // 添加部署名称
.addClasspathResource("processes/first.bpmn") // 从classpath下面加载文件,一次只能加载一个文件。
.deploy();
System.out.println("部署ID:" + deployment.getId());// 1
System.out.println("部署名称:" + deployment.getName());
}
}
(2)查看表:
act_re_deployment 部署表
act_ge_bytearray 资源表
2.5 流程定义启动
(1)启动代码
/** 启动流程实例 */
@Test
public void startProcessInstance() {
// 流程定义的key
String processDefinitionKey = "myProcess";
ProcessInstance pi = runtimeService// 与正在执行的流程实例和执行对象相关的Service
.startProcessInstanceByKey(processDefinitionKey);// 使用流程定义的key启动流程实例,key对应bpmn文件中id的属性值,使用key值启动,默认是按照最新版本的流程定义启动
System.out.println("流程实例ID:" + pi.getId());// 流程实例ID 101
System.out.println("流程定义ID:" + pi.getProcessDefinitionId());// 流程定义ID helloworld:1:4
}
(2)数据库表
详细信息可以查看:act_ru_task表
2.6 查询当前任务
在这儿可以根据人来进行查询,即.taskAssignee(指定个人任务),在设计流程时我们没有指定,所以我们通过指定任务ID来进行查询。
任务ID可以通过表:act_ru_task表
/**查询当前人的个人任务*/
@Test
public void findMyPersonalTask(){
// String assignee = "worker vocation";
List<Task> list = taskService//与正在执行的任务管理相关的Service
.createTaskQuery()//创建任务查询对象
// .taskAssignee(assignee)//指定个人任务查询,指定办理人
.taskId("2505")
.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());
System.out.println("########################################################");
}
}
}
结果:
2.7 结束当前任务
(1)完成代码
/**完成我的任务*/
@Test
public void completeMyPersonalTask(){
//任务ID
String taskId = "2505";
taskService//与正在执行的任务管理相关的Service
.complete(taskId);
System.out.println("完成任务:任务ID:"+taskId);
}
详情:
2.8 结束流程
现在来说,流程才刚刚完成员工的申请,也就是worker vocation。下面是领导审批,也就是leader approval。
其实过程一样,先来查看数据库表:act_hi_actinst
可以查看现在的所有节点,在最后一行数据中,我们就可以发现,该任务正在进行,还没有结束时间。
我们也可以继续查看正在进行的任务表:act_ru_task
将查看任务的代码中ID改成:5002可以查看任务。
将完成额任务的代码中的ID 改成:5002 可以完成任务。
结论:以上完成了 spingboot1.5.10版本和Activiti6.0.0的结合
spingboot2.1.2版本和Activiti6.0.0的结合
均成功~~
配置不同点:
(1)2.1.2版本 添加 @EnableAutoConfiguration(exclude=SecurityAutoConfiguration.class)
(2)数据库时间报错,需要调节Mysql的时间
打开mysql的命令行,添加:
set global time_zone='+8:00'