Activiti工作流引擎基础入门【收藏可做笔记系列】

执行完成后我们查看数据库, 创建了 25 张表,结果如下:

请添加图片描述

到这,我们就完成activiti运行需要的数据库和表的创建。

3.4 表结构介绍


3.4.1 表的命名规则和作用

看到刚才创建的表,我们发现Activiti 的表都以 ACT_ 开头。

第二部分是表示表的用途的两个字母标识。 用途也和服务的 API 对应。

ACT_RE :'RE’表示 repository。 这个前缀的表包含了流程定义和流程静态资源 (图片,规则,等等)。

ACT_RU:'RU’表示 runtime。 这些运行时的表,包含流程实例,任务,变量,异步任务,等运行中的数据。 Activiti 只在流程实例执行过程中保存这些数据, 在流程结束时就会删除这些记录。 这样运行时表可以一直很小速度很快。

ACT_HI:'HI’表示 history。 这些表包含历史数据,比如历史流程实例, 变量,任务等等。

ACT_GE : GE 表示 general。 通用数据, 用于不同场景下

3.4.2 Activiti数据表介绍

| 表分类 | 表名 | 解释 |

| — | — | — |

| 一般数据 | | |

| | [ACT_GE_BYTEARRAY] | 通用的流程定义和流程资源 |

| | [ACT_GE_PROPERTY] | 系统相关属性 |

| 流程历史记录 | | |

| | [ACT_HI_ACTINST] | 历史的流程实例 |

| | [ACT_HI_ATTACHMENT] | 历史的流程附件 |

| | [ACT_HI_COMMENT] | 历史的说明性信息 |

| | [ACT_HI_DETAIL] | 历史的流程运行中的细节信息 |

| | [ACT_HI_IDENTITYLINK] | 历史的流程运行过程中用户关系 |

| | [ACT_HI_PROCINST] | 历史的流程实例 |

| | [ACT_HI_TASKINST] | 历史的任务实例 |

| | [ACT_HI_VARINST] | 历史的流程运行中的变量信息 |

| 流程定义表 | | |

| | [ACT_RE_DEPLOYMENT] | 部署单元信息 |

| | [ACT_RE_MODEL] | 模型信息 |

| | [ACT_RE_PROCDEF] | 已部署的流程定义 |

| 运行实例表 | | |

| | [ACT_RU_EVENT_SUBSCR] | 运行时事件 |

| | [ACT_RU_EXECUTION] | 运行时流程执行实例 |

| | [ACT_RU_IDENTITYLINK] | 运行时用户关系信息,存储任务节点与参与者的相关信息 |

| | [ACT_RU_JOB] | 运行时作业 |

| | [ACT_RU_TASK] | 运行时任务 |

| | [ACT_RU_VARIABLE] | 运行时变量表 |

四、Activiti类关系图

========================================================================

上面我们完成了Activiti数据库表的生成,java代码中我们调用Activiti的工具类,下面来了解Activiti的类关系

4.1 类关系图


请添加图片描述

在新版本中,我们通过实验可以发现IdentityService,FormService两个Serivce都已经删除了。

所以后面我们对于这两个Service也不讲解了,但老版本中还是有这两个Service,同学们需要了解一下

4.2 activiti.cfg.xml


activiti的引擎配置文件,包括:ProcessEngineConfiguration的定义、数据源定义、事务管理器等,此文件其实就是一个spring配置文件。

4.3 流程引擎配置类


流程引擎的配置类(ProcessEngineConfiguration),通过ProcessEngineConfiguration可以创建工作流引擎ProceccEngine,常用的两种方法如下:

4.3.1 StandaloneProcessEngineConfiguration

使用StandaloneProcessEngineConfigurationActiviti可以单独运行,来创建ProcessEngine,Activiti会自己处理事务。

配置文件方式:

通常在activiti.cfg.xml配置文件中定义一个id为 processEngineConfiguration 的bean.

方法如下:

<bean id=“processEngineConfiguration”

class=“org.activiti.engine.impl.cfg.StandaloneProcessEngineConfiguration”>

还可以加入连接池:

<?xml version="1.0" encoding="UTF-8"?>

<beans xmlns=“http://www.springframework.org/schema/beans”

xmlns:xsi=“http://www.w3.org/2001/XMLSchema-instance”

xmlns:context=“http://www.springframework.org/schema/context”

xmlns:tx=“http://www.springframework.org/schema/tx”

xsi:schemaLocation="http://www.springframework.org/schema/beans

http://www.springframework.org/schema/beans/spring-beans.xsd

http://www.springframework.org/schema/contex

http://www.springframework.org/schema/context/spring-context.xsd

http://www.springframework.org/schema/tx

http://www.springframework.org/schema/tx/spring-tx.xsd">

<bean id=“processEngineConfiguration”

class=“org.activiti.engine.impl.cfg.StandaloneProcessEngineConfiguration”>

4.3.2 SpringProcessEngineConfiguration

通过org.activiti.spring.SpringProcessEngineConfiguration 与Spring整合。

创建spring与activiti的整合配置文件:

activity-spring.cfg.xml(名称可修改)

<beans xmlns=“http://www.springframework.org/schema/beans”

xmlns:xsi=“http://www.w3.org/2001/XMLSchema-instance” xmlns:mvc=“http://www.springframework.org/schema/mvc”

xmlns:context=“http://www.springframework.org/schema/context”

xmlns:aop=“http://www.springframework.org/schema/aop” xmlns:tx=“http://www.springframework.org/schema/tx”

xsi:schemaLocation="http://www.springframework.org/schema/beans

http://www.springframework.org/schema/beans/spring-beans-3.1.xsd

http://www.springframework.org/schema/mvc

http://www.springframework.org/schema/mvc/spring-mvc-3.1.xsd

http://www.springframework.org/schema/context

http://www.springframework.org/schema/context/spring-context-3.1.xsd

http://www.springframework.org/schema/aop

http://www.springframework.org/schema/aop/spring-aop-3.1.xsd

http://www.springframework.org/schema/tx

http://www.springframework.org/schema/tx/spring-tx-3.1.xsd ">

<bean id=“repositoryService” factory-bean=“processEngine”

factory-method=“getRepositoryService” />

<bean id=“runtimeService” factory-bean=“processEngine”

factory-method=“getRuntimeService” />

<bean id=“taskService” factory-bean=“processEngine”

factory-method=“getTaskService” />

<bean id=“transactionManager”

class=“org.springframework.jdbc.datasource.DataSourceTransactionManager”>

<tx:advice id=“txAdvice” transaction-manager=“transactionManager”>

tx:attributes</tx:attributes>

<tx:method name=“save*” propagation=“REQUIRED” />

<tx:method name=“insert*” propagation=“REQUIRED” />

<tx:method name=“delete*” propagation=“REQUIRED” />

<tx:method name=“update*” propagation=“REQUIRED” />

<tx:method name=“find*” propagation=“SUPPORTS” read-only=“true” />

<tx:method name=“get*” propagation=“SUPPORTS” read-only=“true” />

</tx:attributes>

</tx:advice>

<aop:config proxy-target-class=“true”>

<aop:advisor advice-ref=“txAdvice” pointcut="execution(* com.itheima.ihrm.service.impl..(…))" />

</aop:config>

创建processEngineConfiguration

ProcessEngineConfiguration configuration = ProcessEngineConfiguration.createProcessEngineConfigurationFromResource(“activiti.cfg.xml”)

​ 上边的代码要求activiti.cfg.xml中必须有一个processEngineConfiguration的bean

也可以使用下边的方法,更改bean 的名字:

ProcessEngineConfiguration.createProcessEngineConfigurationFromResource(String resource, String beanName);

4.4 工作流引擎创建


工作流引擎(ProcessEngine),相当于一个门面接口,通过ProcessEngineConfiguration创建processEngine,通过ProcessEngine创建各个service接口。

4.4.1 默认创建方式

将activiti.cfg.xml文件名及路径固定,且activiti.cfg.xml文件中有 processEngineConfiguration的配置, 可以使用如下代码创建processEngine:

//直接使用工具类 ProcessEngines,使用classpath下的activiti.cfg.xml中的配置创建processEngine

ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();

System.out.println(processEngine);

4.4.2 一般创建方式

//先构建ProcessEngineConfiguration

ProcessEngineConfiguration configuration = ProcessEngineConfiguration.createProcessEngineConfigurationFromResource(“activiti.cfg.xml”);

//通过ProcessEngineConfiguration创建ProcessEngine,此时会创建数据库

ProcessEngine processEngine = configuration.buildProcessEngine();

4.5 Servcie服务接口


Service是工作流引擎提供用于进行工作流部署、执行、管理的服务接口,我们使用这些接口可以就是操作服务对应的数据表

4.5.1 Service创建方式

通过ProcessEngine创建Service

方式如下:

RuntimeService runtimeService = processEngine.getRuntimeService();

RepositoryService repositoryService = processEngine.getRepositoryService();

TaskService taskService = processEngine.getTaskService();

4.5.2 Service总览

| service名称 | service作用 |

| — | — |

| RepositoryService | activiti的资源管理类 |

| RuntimeService | activiti的流程运行管理类 |

| TaskService | activiti的任务管理类 |

| HistoryService | activiti的历史管理类 |

| ManagerService | activiti的引擎管理类 |

简单介绍:

RepositoryService

是activiti的资源管理类,提供了管理和控制流程发布包和流程定义的操作。使用工作流建模工具设计的业务流程图需要使用此service将流程定义文件的内容部署到计算机。

除了部署流程定义以外还可以:查询引擎中的发布包和流程定义。

暂停或激活发布包,对应全部和特定流程定义。 暂停意味着它们不能再执行任何操作了,激活是对应的反向操作。获得多种资源,像是包含在发布包里的文件, 或引擎自动生成的流程图。

获得流程定义的pojo版本, 可以用来通过java解析流程,而不必通过xml。

RuntimeService

Activiti的流程运行管理类。可以从这个服务类中获取很多关于流程执行相关的信息

TaskService

Activiti的任务管理类。可以从这个类中获取任务的信息。

HistoryService

Activiti的历史管理类,可以查询历史信息,执行流程时,引擎会保存很多数据(根据配置),比如流程实例启动时间,任务的参与者, 完成任务的时间,每个流程实例的执行路径,等等。 这个服务主要通过查询功能来获得这些数据。

ManagementService

Activiti的引擎管理类,提供了对 Activiti 流程引擎的管理和维护功能,这些功能不在工作流驱动的应用程序中使用,主要用于 Activiti 系统的日常维护。

五、Activiti入门

======================================================================

在本章内容中,我们来创建一个Activiti工作流,并启动这个流程。

创建Activiti工作流主要包含以下几步:

1、定义流程,按照BPMN的规范,使用流程定义工具,用流程符号把整个流程描述出来

2、部署流程,把画好的流程定义文件,加载到数据库中,生成表的数据

3、启动流程,使用java代码来操作数据库表中的内容

5.1 流程符号


BPMN 2.0是业务流程建模符号2.0的缩写。

它由Business Process Management Initiative这个非营利协会创建并不断发展。作为一种标识,BPMN 2.0是使用一些符号来明确业务流程设计流程图的一整套符号规范,它能增进业务建模时的沟通效率。

目前BPMN2.0是最新的版本,它用于在BPM上下文中进行布局和可视化的沟通。

接下来我们先来了解在流程设计中常见的 符号。

BPMN2.0的基本符合主要包含:

事件 Event

请添加图片描述

活动 Activity

活动是工作或任务的一个通用术语。一个活动可以是一个任务,还可以是一个当前流程的子处理流程; 其次,你还可以为活动指定不同的类型。常见活动如下:

请添加图片描述

网关 GateWay

网关用来处理决策,有几种常用网关需要了解:

请添加图片描述

排他网关 (x)

——只有一条路径会被选择。流程执行到该网关时,按照输出流的顺序逐个计算,当条件的计算结果为true时,继续执行当前网关的输出流;

​ 如果多条线路计算结果都是 true,则会执行第一个值为 true 的线路。如果所有网关计算结果没有true,则引擎会抛出异常。

​ 排他网关需要和条件顺序流结合使用,default 属性指定默认顺序流,当所有的条件不满足时会执行默认顺序流。

并行网关 (+)

——所有路径会被同时选择

​ 拆分 —— 并行执行所有输出顺序流,为每一条顺序流创建一个并行执行线路。

​ 合并 —— 所有从并行网关拆分并执行完成的线路均在此等候,直到所有的线路都执行完成才继续向下执行。

包容网关 (+)

—— 可以同时执行多条线路,也可以在网关上设置条件

​ 拆分 —— 计算每条线路上的表达式,当表达式计算结果为true时,创建一个并行线路并继续执行

​ 合并 —— 所有从并行网关拆分并执行完成的线路均在此等候,直到所有的线路都执行完成才继续向下执行。

事件网关 (+)

—— 专门为中间捕获事件设置的,允许设置多个输出流指向多个不同的中间捕获事件。当流程执行到事件网关后,流程处于等待状态,需要等待抛出事件才能将等待状态转换为活动状态。

流向 Flow

流是连接两个流程节点的连线。常见的流向包含以下几种:

请添加图片描述

5.2 流程设计器使用


Activiti-Designer使用

Palette(画板)

在idea中安装插件即可使用,画板中包括以下结点:

Connection—连接

Event—事件

Task—任务

Gateway—网关

Container—容器

Boundary event—边界事件

Intermediate event- -中间事件

流程图设计完毕保存生成.bpmn文件

新建流程(IDEA工具)

首先选中存放图形的目录(选择resources下的bpmn目录),点击菜单:New -> BpmnFile,如图:

请添加图片描述

弹出如下图所示框,输入evection 表示 出差审批流程:

请添加图片描述

起完名字evection后(默认扩展名为bpmn),就可以看到流程设计页面,如图所示:

请添加图片描述

左侧区域是绘图区,右侧区域是palette画板区域

鼠标先点击画板的元素即可在左侧绘图

绘制流程

使用滑板来绘制流程,通过从右侧把图标拖拽到左侧的画板,最终效果如下:

请添加图片描述

指定流程定义Key

流程定义key即流程定义的标识,通过properties视图查看流程的key

请添加图片描述

指定任务负责人

在properties视图指定每个任务结点的负责人,如:填写出差申请的负责人为 zhangsan

请添加图片描述

经理审批负责人为 jerry

总经理审批负责人为 jack

财务审批负责人为 rose

六、流程操作

================================================================

6.1 流程定义


概述

流程定义是线下按照bpmn2.0标准去描述 业务流程,通常使用idea中的插件对业务流程进行建模。

使用idea下的designer设计器绘制流程,并会生成两个文件:.bpmn和.png

.bpmn文件

使用activiti-desinger设计业务流程,会生成.bpmn文件,上面我们已经创建好了bpmn文件

BPMN 2.0根节点是definitions节点。 这个元素中,可以定义多个流程定义(不过我们建议每个文件只包含一个流程定义, 可以简化开发过程中的维护难度)。 注意,definitions元素 最少也要包含xmlns 和 targetNamespace的声明。 targetNamespace可以是任意值,它用来对流程实例进行分类。

流程定义部分:定义了流程每个结点的描述及结点之间的流程流转。

流程布局定义:定义流程每个结点在流程图上的位置坐标等信息。

生成.png图片文件

IDEA工具中的操作方式

1、修改文件后缀为xml

首先将evection.bpmn文件改名为evection.xml,如下图:

请添加图片描述

evection.xml修改前的bpmn文件,效果如下:

请添加图片描述

2、使用designer设计器打开.xml文件

在evection.xml文件上面,点右键并选择Diagrams菜单,再选择Show BPMN2.0 Designer…

请添加图片描述

3、查看打开的文件

打开后,却出现乱码,如图:

请添加图片描述

4、解决中文乱码

1、打开Settings,找到File Encodings,把encoding的选项都选择UTF-8

请添加图片描述

2、打开IDEA安装路径,找到如下的安装目录

请添加图片描述

根据自己所安装的版本来决定,我使用的是64位的idea,所以在idea64.exe.vmoptions文件的最后一行追加一条命令: -Dfile.encoding=UTF-8

如下所示:

请添加图片描述

一定注意,不要有空格,否则重启IDEA时会打不开,然后 重启IDEA。

如果以上方法已经做完,还出现乱码,就再修改一个文件,并在文件的末尾添加: -Dfile.encoding=UTF-8,然后重启idea,如图:

请添加图片描述

最后重新在evection.xml文件上面,点右键并选择Diagrams菜单,再选择Show BPMN2.0 Designer…,看到生成图片,如图:

请添加图片描述

到此,解决乱码问题

5、导出为图片文件

点击Export To File的小图标,打开如下窗口,注意填写文件名及扩展名,选择好保存图片的位置:

请添加图片描述

然后,我们把png文件拷贝到resources下的bpmn目录,并且把evection.xml改名为evection.bpmn。

6.2 流程定义部署


概述

将上面在设计器中定义的流程部署到activiti数据库中,就是流程定义部署。

通过调用activiti的api将流程定义的bpmn和png两个文件一个一个添加部署到activiti中,也可以将两个文件打成zip包进行部署。

单个文件部署方式

分别将bpmn文件和png图片文件部署。

package com.itheima.test;

import org.activiti.engine.ProcessEngine;

import org.activiti.engine.ProcessEngines;

import org.activiti.engine.RepositoryService;

import org.activiti.engine.repository.Deployment;

import org.junit.Test;

public class ActivitiDemo {

/**

  • 部署流程定义

*/

@Test

public void testDeployment(){

// 1、创建ProcessEngine

ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();

// 2、得到RepositoryService实例

RepositoryService repositoryService = processEngine.getRepositoryService();

// 3、使用RepositoryService进行部署

Deployment deployment = repositoryService.createDeployment()

.addClasspathResource(“bpmn/evection.bpmn”) // 添加bpmn资源

.addClasspathResource(“bpmn/evection.png”) // 添加png资源

.name(“出差申请流程”)

.deploy();

// 4、输出部署信息

System.out.println(“流程部署id:” + deployment.getId());

System.out.println(“流程部署名称:” + deployment.getName());

}

}

执行此操作后activiti会将上边代码中指定的bpm文件和图片文件保存在activiti数据库。

压缩包部署方式

将evection.bpmn和evection.png压缩成zip包。

@Test

public void deployProcessByZip() {

// 定义zip输入流

InputStream inputStream = this

.getClass()

.getClassLoader()

.getResourceAsStream(

“bpmn/evection.zip”);

ZipInputStream zipInputStream = new ZipInputStream(inputStream);

// 获取repositoryService

RepositoryService repositoryService = processEngine

.getRepositoryService();

// 流程部署

Deployment deployment = repositoryService.createDeployment()

.addZipInputStream(zipInputStream)

.deploy();

System.out.println(“流程部署id:” + deployment.getId());

System.out.println(“流程部署名称:” + deployment.getName());

}

执行此操作后activiti会将上边代码中指定的bpm文件和图片文件保存在activiti数据库。

操作数据表

流程定义部署后操作activiti的3张表如下:

act_re_deployment 流程定义部署表,每部署一次增加一条记录

act_re_procdef 流程定义表,部署每个新的流程定义都会在这张表中增加一条记录

act_ge_bytearray 流程资源表

接下来我们来看看,写入了什么数据:

SELECT * FROM act_re_deployment #流程定义部署表,记录流程部署信息

结果:

请添加图片描述

SELECT * FROM act_re_procdef #流程定义表,记录流程定义信息

结果:

注意,KEY 这个字段是用来唯一识别不同流程的关键字

请添加图片描述

SELECT * FROM act_ge_bytearray #资源表

结果:

请添加图片描述

注意:

act_re_deployment和act_re_procdef一对多关系,一次部署在流程部署表生成一条记录,但一次部署可以部署多个流程定义,每个流程定义在流程定义表生成一条记录。每一个流程定义在act_ge_bytearray会存在两个资源记录,bpmn和png。

建议:一次部署一个流程,这样部署表和流程定义表是一对一有关系,方便读取流程部署及流程定义信息。

6.3 启动流程实例


流程定义部署在activiti后就可以通过工作流管理业务流程了,也就是说上边部署的出差申请流程可以使用了。

针对该流程,启动一个流程表示发起一个新的出差申请单,这就相当于java类与java对象的关系,类定义好后需要new创建一个对象使用,当然可以new多个对象。对于请出差申请流程,张三发起一个出差申请单需要启动一个流程实例,出差申请单发起一个出差单也需要启动一个流程实例。

代码如下:

/**

  • 启动流程实例

*/

@Test

public void testStartProcess(){

// 1、创建ProcessEngine

ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();

// 2、获取RunTimeService

RuntimeService runtimeService = processEngine.getRuntimeService();

// 3、根据流程定义Id启动流程

ProcessInstance processInstance = runtimeService

.startProcessInstanceByKey(“myEvection”);

// 输出内容

System.out.println(“流程定义id:” + processInstance.getProcessDefinitionId());

System.out.println(“流程实例id:” + processInstance.getId());

System.out.println(“当前活动Id:” + processInstance.getActivityId());

}

输出内容如下:

请添加图片描述

操作数据表

act_hi_actinst 流程实例执行历史

act_hi_identitylink 流程的参与用户历史信息

act_hi_procinst 流程实例历史信息

act_hi_taskinst 流程任务历史信息

act_ru_execution 流程执行信息

act_ru_identitylink 流程的参与用户信息

act_ru_task 任务信息

6.4 任务查询


流程启动后,任务的负责人就可以查询自己当前需要处理的任务,查询出来的任务都是该用户的待办任务。

/**

  • 查询当前个人待执行的任务

*/

@Test

public void testFindPersonalTaskList() {

// 任务负责人

String assignee = “zhangsan”;

ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();

// 创建TaskService

TaskService taskService = processEngine.getTaskService();

// 根据流程key 和 任务负责人 查询任务

List list = taskService.createTaskQuery()

.processDefinitionKey(“myEvection”) //流程Key

.taskAssignee(assignee)//只查询该任务负责人的任务

.list();

for (Task task : list) {

System.out.println(“流程实例id:” + task.getProcessInstanceId());

System.out.println(“任务id:” + task.getId());

System.out.println(“任务负责人:” + task.getAssignee());

System.out.println(“任务名称:” + task.getName());

}

}

输出结果如下:

流程实例id:2501

任务id:2505

任务负责人:zhangsan

任务名称:创建出差申请

6.5 流程任务处理


任务负责人查询待办任务,选择任务进行处理,完成任务。

// 完成任务

@Test

public void completTask(){

// 获取引擎

ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();

// 获取taskService

TaskService taskService = processEngine.getTaskService();

// 根据流程key 和 任务的负责人 查询任务

// 返回一个任务对象

Task task = taskService.createTaskQuery()

.processDefinitionKey(“myEvection”) //流程Key

.taskAssignee(“zhangsan”) //要查询的负责人

.singleResult();

// 完成任务,参数:任务id

taskService.complete(task.getId());

}

注意:这里使用的是.singleResult();,此时就必须保证数据库中唯一存在,不然就需要使用.list()来返回一个List集合,遍历集合再进行信息的获取。下面使用到.singleResult();的地方同理。

6.6 流程定义信息查询


查询流程相关信息,包含流程定义,流程部署,流程定义版本

/**

  • 查询流程定义

*/

@Test

public void queryProcessDefinition(){

// 获取引擎

ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();

// repositoryService

RepositoryService repositoryService = processEngine.getRepositoryService();

// 得到ProcessDefinitionQuery 对象

ProcessDefinitionQuery processDefinitionQuery = repositoryService.createProcessDefinitionQuery();

// 查询出当前所有的流程定义

// 条件:processDefinitionKey =evection

// orderByProcessDefinitionVersion 按照版本排序

// desc倒叙

// list 返回集合

List definitionList = processDefinitionQuery.processDefinitionKey(“myEvection”)

.orderByProcessDefinitionVersion()

.desc()

.list();

// 输出流程定义信息

for (ProcessDefinition processDefinition : definitionList) {

System.out.println(“流程定义 id=”+processDefinition.getId());

System.out.println(“流程定义 name=”+processDefinition.getName());

System.out.println(“流程定义 key=”+processDefinition.getKey());

System.out.println(“流程定义 Version=”+processDefinition.getVersion());

System.out.println(“流程部署ID =”+processDefinition.getDeploymentId());

}

}

输出结果:

流程定义id:myEvection:1:4

流程定义名称:出差申请单

流程定义key:myEvection

流程定义版本:1

6.7 流程删除


public void deleteDeployment() {

// 流程部署id

String deploymentId = “1”;

ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();

// 通过流程引擎获取repositoryService

RepositoryService repositoryService = processEngine

.getRepositoryService();

//删除流程定义,如果该流程定义已有流程实例启动则删除时出错

repositoryService.deleteDeployment(deploymentId);

//设置true 级联删除流程定义,即使该流程有流程实例启动也可以删除,设置为false非级别删除方式,如果流程

//repositoryService.deleteDeployment(deploymentId, true);

}

说明:

使用repositoryService删除流程定义,历史表信息不会被删除

如果该流程定义下没有正在运行的流程,则可以用普通删除。

如果该流程定义下存在已经运行的流程,使用普通删除报错,可用级联删除方法将流程及相关记录全部删除。

先删除没有完成流程节点,最后就可以完全删除流程定义信息

项目开发中级联删除操作一般只开放给超级管理员使用.

6.8 流程资源下载


现在我们的流程资源文件已经上传到数据库了,如果其他用户想要查看这些资源文件,可以从数据库中把资源文件下载到本地。

解决方案有:

1、jdbc对blob类型,clob类型数据读取出来,保存到文件目录

2、使用activiti的api来实现

使用commons-io.jar 解决IO的操作

引入commons-io依赖包

commons-io

commons-io

2.6

通过流程定义对象获取流程定义资源,获取bpmn和png

import org.apache.commons.io.IOUtils;

@Test

public void deleteDeployment(){

// 获取引擎

ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();

// 获取repositoryService

RepositoryService repositoryService = processEngine.getRepositoryService();

// 根据部署id 删除部署信息,如果想要级联删除,可以添加第二个参数,true

repositoryService.deleteDeployment(“1”);

}

public void queryBpmnFile() throws IOException {

// 1、得到引擎

ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();

// 2、获取repositoryService

RepositoryService repositoryService = processEngine.getRepositoryService();

// 3、得到查询器:ProcessDefinitionQuery,设置查询条件,得到想要的流程定义

ProcessDefinition processDefinition = repositoryService.createProcessDefinitionQuery()

.processDefinitionKey(“myEvection”)

.singleResult();

// 4、通过流程定义信息,得到部署ID

String deploymentId = processDefinition.getDeploymentId();

// 5、通过repositoryService的方法,实现读取图片信息和bpmn信息

// png图片的流

InputStream pngInput = repositoryService.getResourceAsStream(deploymentId, processDefinition.getDiagramResourceName());

// bpmn文件的流

InputStream bpmnInput = repositoryService.getResourceAsStream(deploymentId, processDefinition.getResourceName());

// 6、构造OutputStream流

File file_png = new File(“d:/evectionflow01.png”);

File file_bpmn = new File(“d:/evectionflow01.bpmn”);

FileOutputStream bpmnOut = new FileOutputStream(file_bpmn);

FileOutputStream pngOut = new FileOutputStream(file_png);

// 7、输入流,输出流的转换

IOUtils.copy(pngInput,pngOut);

IOUtils.copy(bpmnInput,bpmnOut);

// 8、关闭流

pngOut.close();

bpmnOut.close();

pngInput.close();

bpmnInput.close();

}

说明:

  1. deploymentId为流程部署ID

  2. resource_name为act_ge_bytearray表中NAME_列的值

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

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

最后的将输入流中的图片资源进行输出。

6.9 流程历史信息的查看


即使流程定义已经删除了,流程执行的历史信息通过前面的分析,依然保存在activiti的act_hi_*相关的表中。所以我们还是可以查询流程执行的历史信息,可以通过HistoryService来查看相关的历史记录。

/**

  • 查看历史信息

*/

@Test

public void findHistoryInfo(){

// 获取引擎

ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();

// 获取HistoryService

HistoryService historyService = processEngine.getHistoryService();

// 获取 actinst表的查询对象

HistoricActivityInstanceQuery instanceQuery = historyService.createHistoricActivityInstanceQuery();

// 查询 actinst表,条件:根据 InstanceId 查询

// instanceQuery.processInstanceId(“2501”);

// 查询 actinst表,条件:根据 DefinitionId 查询

instanceQuery.processDefinitionId(“myEvection:1:4”);

// 增加排序操作,orderByHistoricActivityInstanceStartTime 根据开始时间排序 asc 升序

instanceQuery.orderByHistoricActivityInstanceStartTime().asc();

// 查询所有内容

List activityInstanceList = instanceQuery.list();

// 输出

for (HistoricActivityInstance hi : activityInstanceList) {

System.out.println(hi.getActivityId());

System.out.println(hi.getActivityName());

System.out.println(hi.getProcessDefinitionId());

System.out.println(hi.getProcessInstanceId());

System.out.println(“<==========================>”);

}

}

文末

技术是没有终点的,也是学不完的,最重要的是活着、不秃。

零基础入门的时候看书还是看视频,我觉得成年人,何必做选择题呢,两个都要。喜欢看书就看书,喜欢看视频就看视频。

最重要的是在自学的过程中,一定不要眼高手低,要实战,把学到的技术投入到项目当中,解决问题,之后进一步锤炼自己的技术。

自学最怕的就是缺乏自驱力,一定要自律,杜绝“三天打鱼两天晒网”,到最后白忙活一场。

高度自律的同时,要保持耐心,不抛弃不放弃,切勿自怨自艾,每天给自己一点点鼓励,学习的劲头就会很足,不容易犯困。

技术学到手后,找工作的时候一定要好好准备一份简历,不要无头苍蝇一样去海投简历,容易“竹篮打水一场空”。好好的准备一下简历,毕竟是找工作的敲门砖。

拿到面试邀请后,在面试的过程中一定要大大方方,尽力把自己学到的知识舒适地表达出来,不要因为是自学就不够自信,给面试官一个好的印象,面试成功的几率就会大很多,加油吧,骚年!

  • 7
    点赞
  • 28
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值