背景
应用flowable自带的flowable-ui制作流程图
应用springboot开发流程应用的接口实现流程的业务性能
flowable-ui部署运行
flowable-6.6.0 运行 官网demo
参考文档:
https://flowable.com/open-sou…
1、从官网下载flowable-6.6.0 : https://github.com/flowable/f…
2、将压缩包中的 flowable-6.6.0\wars\flowable-ui.war 丢到Tomcat中跑起来
3、关上http://localhost:8080/flowabl… 用账户:admin/test 登录
4、进入APP.MODELER创立流程,之后能够导出流程到我的项目中应用,或者配置
apache-tomcat-9.0.37\webapps\flowable-ui\WEB-INF\classes\flowable-default.properties连贯本地数据库
留神:须要将java驱动jar(mysql-connector-java-5.1.45.jar)复制到 apache-tomcat-9.0.37\webapps\flowable-rest\WEB-INF\lib这样创立的流程后端程序就能间接应用
绘制流程图
依据业务须要在 flowable-ui>APP.MODELER外面绘制流程图,示例如上图。先解释一些概念。
事件(event)通常用于为流程生命周期中产生的事件建模,图里是【开始、完结】两个圈。
程序流(sequence flow)是流程中两个元素间的连接器。图里是【箭头线段】。
网关(gateway)用于管制执行的流向。图里是【菱形(两头有X)】
用户工作(user task)用于对须要人工执行的工作进行建模。图里是【矩形】。
简略的工作流大略就这些元素(还有很多这里就不扩大了)。上面形容一下工作流是如何流动的。
首先启动了工作流后,由【开始】节点主动流向【学生】节点,期待该工作执行。工作被调配的学生用户执行后流向 【老师】节点,再次期待该工作执行。被调配的老师用户执行后流向 【网关】,网关以此查看每个进口,流向符合条件的工作,比方这里老师执行工作时是批准,就流向【校长】节点,期待该工作执行。执行后跟老师相似,批准后就流向【完结】节点,整个流程到此结束。
绘图细节:
1、保留流程模型
2、程序流能够设置流条件来限度流动,比方下面的网关进口就设置了条件
3、工作须要分配任务的执行用户,能够调配到候选组,也能够间接调配到候选人
最初导出工作流文件
文件内容
<?xml version="1.0" encoding="UTF-8"?>
<definitions xmlns="http://www.omg.org/spec/BPMN/20100524/MODEL" xmlns:xsi="http://www.w3.org/2001/XMLSchema-insmtece" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:flowable="http://flowable.org/bpmn" xmlns:bpmndi="http://www.omg.org/spec/BPMN/20100524/DI" xmlns:omgdc="http://www.omg.org/spec/DD/20100524/DC" xmlns:omgdi="http://www.omg.org/spec/DD/20100524/DI" typeLanguage="http://www.w3.org/2001/XMLSchema" expressionLanguage="http://www.w3.org/1999/XPath" targetNamespace="http://www.flowable.org/processdef">
<process id="leave_approval" name="销假审批" isExecutable="true">
<startEvent id="start" name="开始" flowable:initiator="startuser" flowable:formFieldValidation="true"></startEvent>
<userTask id="stu_task" name="学生" flowable:candidateGroups="stu_group" flowable:formFieldValidation="true"></userTask>
<sequenceFlow id="flow1" sourceRef="start" targetRef="stu_task"></sequenceFlow>
<userTask id="te_task" name="老师" flowable:candidateGroups="te_group" flowable:formFieldValidation="true"></userTask>
<exclusiveGateway id="getway1" name="网关1"></exclusiveGateway>
<userTask id="mte_task" name="校长" flowable:candidateGroups="mte_group" flowable:formFieldValidation="true"></userTask>
<exclusiveGateway id="getway2" name="网关2"></exclusiveGateway>
<endEvent id="end" name="完结"></endEvent>
<sequenceFlow id="flow1" name="销假" sourceRef="stu_task" targetRef="te_task" skipExpression="${command=='agree'}"></sequenceFlow>
<sequenceFlow id="flow3_1" name="批准" sourceRef="getway1" targetRef="mte_task">
<conditionExpression xsi:type="tFormalExpression"><![CDATA[${command=='agree'}]]></conditionExpression>
</sequenceFlow>
<sequenceFlow id="flow2" name="审批" sourceRef="te_task" targetRef="getway1"></sequenceFlow>
<sequenceFlow id="flow3_2" name="回绝" sourceRef="getway1" targetRef="stu_task">
<conditionExpression xsi:type="tFormalExpression"><![CDATA[${command=='refuse'}]]></conditionExpression>
</sequenceFlow>
<sequenceFlow id="flow4" name="审批" sourceRef="mte_task" targetRef="getway2"></sequenceFlow>
<sequenceFlow id="flow4_1" name="批准" sourceRef="getway2" targetRef="end" skipExpression="${command=='free'}">
<conditionExpression xsi:type="tFormalExpression"><![CDATA[${command=='agree'}]]></conditionExpression>
</sequenceFlow>
<sequenceFlow id="flow4_2" name="回绝" sourceRef="getway2" targetRef="stu_task">
<conditionExpression xsi:type="tFormalExpression"><![CDATA[${command=='refuse'}]]></conditionExpression>
</sequenceFlow>
</process>
<bpmndi:BPMNDiagram id="BPMNDiagram_leave_approval">
这里先省略
</bpmndi:BPMNDiagram>
</definitions>
【残缺文件】:leave_approval.bpmn20.xml
4、bpmn文件导入
如果须要,能够把这个流程文件下载下来,间接导入应用
后盾我的项目搭建
后盾我的项目基于jdk8,应用springboot框架
Spring Boot 根底就不介绍了,举荐下这个实战教程:
https://github.com/javastacks…
spring 版本
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>2.3.0.RELEASE</version>
<relativePath/> <!-- lookup parent from repository -->
</parent>
我的项目依赖pom.xml
<dependency>
<groupId>org.flowable</groupId>
<artifactId>flowable-spring-boot-starter</artifactId>
<version>6.6.0</version>
</dependency>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>5.1.45</version>
</dependency>
我的项目配置application.yml
spring:
datasource:
url: jdbc:mysql://localhost:3306/flowable?useSSL=false&characterEncoding=UTF-8&serverTimezone=GMT%2B8
driver-class-name: com.mysql.jdbc.Driver
username: root
password: 123456
数据库
1、Flowable的所有数据库表都以ACT_结尾。第二局部是阐明表用处的两字符标示符。服务API的命名也大略合乎这个规定。
2、ACT_RE_: ‘RE’代表repository。带有这个前缀的表蕴含“动态”信息,例如流程定义与流程资源(图片、规定等)。
3、ACT_RU_: ‘RU’代表runtime。这些表存储运行时信息,例如流程实例(process instance)、用户工作(user task)、变量(variable)、作业(job)等。Flowable只在流程实例运行中保留运行时数据,并在流程实例完结时删除记录。这样保障运行时表小和快。
4、ACT_HI_: ‘HI’代表history。这些表存储历史数据,例如已实现的流程实例、变量、工作等。
5、ACT_GE_: 通用数据。在多处应用。
1)通用数据表(2个)
- act_ge_bytearray:二进制数据表,如流程定义、流程模板、流程图的字节流文件;
- act_ge_property:属性数据表(不罕用);
2)历史表(8个,HistoryService接口操作的表)
- act_hi_actinst:历史节点表,寄存流程实例运行的各个节点信息(蕴含开始、完结等非工作节点);
- act_hi_attachment:历史附件表,寄存历史节点上传的附件信息(不罕用);
- act_hi_comment:历史意见表;
- act_hi_detail:历史详情表,存储节点运行的一些信息(不罕用);
- act_hi_identitylink:历史流程人员表,存储流程各节点候选、办理人员信息,罕用于查问某人或部门的已办工作;
- act_hi_procinst:历史流程实例表,存储流程实例历史数据(蕴含正在运行的流程实例);
- act_hi_taskinst:历史流程工作表,存储历史工作节点;
- act_hi_varinst:流程历史变量表,存储流程历史节点的变量信息;
3)用户相干表(4个,IdentityService接口操作的表)
- act_id_group:用户组信息表,对应节点选定候选组信息;
- act_id_info:用户扩大信息表,存储用户扩大信息;
- act_id_membership:用户与用户组关系表;
- act_id_user:用户信息表,对应节点选定办理人或候选人信息;
4)流程定义、流程模板相干表(3个,RepositoryService接口操作的表)
- act_re_deployment:部属信息表,存储流程定义、模板部署信息;
- act_re_procdef:流程定义信息表,存储流程定义相干形容信息,但其真正内容存储在act_ge_bytearray表中,以字节模式存储;
- act_re_model:流程模板信息表,存储流程模板相干形容信息,但其真正内容存储在act_ge_bytearray表中,以字节模式存储;
5)流程运行时表(6个,RuntimeService接口操作的表)
- act_ru_task:运行时流程工作节点表,存储运行中流程的工作节点信息,重要,罕用于查问人员或部门的待办工作时应用;
- act_ru_event_subscr:监听信息表,不罕用;
- act_ru_execution:运行时流程执行实例表,记录运行中流程运行的各个分支信息(当没有子流程时,其数据与act_ru_task表数据是一一对应的);
- act_ru_identitylink:运行时流程人员表,重要,罕用于查问人员或部门的待办工作时应用;
- act_ru_job:运行时定时工作数据表,存储流程的定时工作信息;
- act_ru_variable:运行时流程变量数据表,存储运行中的流程各节点的变量信息;
流程引擎API与服务
引擎API是与Flowable交互的最罕用伎俩。总入口点是ProcessEngine。
1、RepositoryService很可能是应用Flowable引擎要用的第一个服务。这个服务提供了治理与管制部署(deployments)与流程定义(process definitions)的操作。治理动态信息,
2、RuntimeService用于启动流程定义的新流程实例。
3、IdentityService很简略。它用于治理(创立,更新,删除,查问……)组与用户。
4、FormService是可选服务。也就是说Flowable没有它也能很好地运行,而不用就义任何性能。
5、HistoryService裸露Flowable引擎收集的所有历史数据。要提供查问历史数据的能力。
6、ManagementService通常在用Flowable编写用户利用时不须要应用。它能够读取数据库表与表原始数据的信息,也提供了对作业(job)的查问与治理操作。
7、DynamicBpmnService可用于批改流程定义中的局部内容,而不须要重新部署它。例如能够批改流程定义中一个用户工作的办理人设置,或者批改一个服务工作中的类名。
接下来应用之前的销假流程图,上代码
import lombok.extern.slf4j.Slf4j;
import org.flowable.engine.HistoryService;
import org.flowable.engine.RepositoryService;
import org.flowable.engine.RuntimeService;
import org.flowable.engine.history.HistoricProcessInstance;
import org.flowable.engine.repository.Deployment;
import org.flowable.engine.repository.ProcessDefinition;
import org.flowable.engine.runtime.Execution;
import org.flowable.engine.runtime.ProcessInstance;
import org.flowable.idm.api.Group;
import org.flowable.idm.api.User;
import org.flowable.task.api.Task;
import org.flowable.task.api.history.HistoricTaskInstance;
import org.springframework.beans.factory.annotation.Autowired;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.zip.ZipInputStream;
/**
* TestFlowable
*
* @Author
* @Date: 2021/10/17 23:35
* @Version 1.0
*/
@Slf4j
public class TestFlowable {
@Autowired
private RepositoryService repositoryService;
@Autowired
private RuntimeService runtimeService;
@Autowired
private HistoryService historyService;
@Autowired
private org.flowable.engine.TaskService taskService;
@Autowired
private org.flowable.engine.IdentityService identityService;
public void createDeploymentZip() {
/*
* @Date: 2021/10/17 23:38
* Step 1: 部署xml(压缩到zip模式,间接xml须要配置相对路径,麻烦,暂不必)
*/
try {
File zipTemp = new File("f:/leave_approval.bpmn20.zip");
ZipInputStream zipInputStream = new ZipInputStream(new FileInputStream(zipTemp));
Deployment deployment = repositoryService
.createDeployment()
.addZipInputStream(zipInputStream)
.deploy();
log.info("部署胜利:{}", deployment.getId());
} catch (FileNotFoundException e) {
e.printStackTrace();
}
/*
* @Date: 2021/10/17 23:40
* Step 2: 查问部署的流程定义
*/
List<ProcessDefinition> list = repositoryService.createProcessDefinitionQuery().processDefinitionKey("leave_approval").list();
List<ProcessDefinition> pages = repositoryService.createProcessDefinitionQuery().processDefinitionKey("leave_approval").listPage(1, 30);
/*
* @Date: 2021/10/17 23:40
* Step 3: 启动流程,创立实例
*/
String processDefinitionKey = "leave_approval";//流程定义的key,对应销假的流程图
String businessKey = "schoolleave";//业务代码,依据本人的业务用
Map<String, Object> variablesDefinition = new HashMap<>();//流程变量,能够自定义裁减
ProcessInstance processInstance = runtimeService.startProcessInstanceByKey(processDefinitionKey, businessKey, variablesDefinition);
log.info("启动胜利:{}", processInstance.getId());
/*
* @Date: 2021/10/17 23:40
* Step 4: 查问指定流程所有启动的实例列表
* 列表,或 分页 删除
*/
List<Execution> executions = runtimeService.createExecutionQuery().processDefinitionKey("leave_approval").list();
List<Execution> executionPages = runtimeService.createExecutionQuery().processDefinitionKey("leave_approval").listPage(1, 30);
// runtimeService.deleteProcessInstance(processInstanceId, deleteReason); //删除实例
/*
* @Date: 2021/10/17 23:40
* Step 5: 学生查问能够操作的工作,并实现工作
*/
String candidateGroup = "stu_group"; //候选组 xml文件外面的 flowable:candidateGroups="stu_group"
List<Task> taskList = taskService.createTaskQuery().taskCandidateGroup(candidateGroup).orderByTaskCreateTime().desc().list();
for (Task task : taskList) {
// 申领工作
taskService.claim(task.getId(), "my");
// 实现
taskService.complete(task.getId());
}
/*
* @Date: 2021/10/17 23:40
* Step 6: 老师查问能够操作的工作,并实现工作
*/
String candidateGroupTe = "te_group"; //候选组 xml文件外面的 flowable:candidateGroups="te_group"
List<Task> taskListTe = taskService.createTaskQuery().taskCandidateGroup(candidateGroupTe).orderByTaskCreateTime().desc().list();
for (Task task : taskListTe) {
// 申领工作
taskService.claim(task.getId(), "myte");
// 实现
Map<String, Object> variables = new HashMap<>();
variables.put("command","agree"); //携带变量,用于网关流程的条件断定,这里的条件是批准
taskService.complete(task.getId(), variables);
}
/*
* @Date: 2021/10/18 0:17
* Step 7: 历史查问,因为一旦流程执行结束,流动的数据都会被清空,下面查问的接口都查不到数据,然而提供历史查问接口
*/
// 历史流程实例
List<HistoricProcessInstance> historicProcessList = historyService.createHistoricProcessInstanceQuery().processDefinitionKey("leave_approval").list();
// 历史工作
List<HistoricTaskInstance> historicTaskList = historyService.createHistoricTaskInstanceQuery().processDefinitionKey("leave_approval").list();
// 实例历史变量 , 工作历史变量
// historyService.createHistoricVariableInstanceQuery().processInstanceId(processInstanceId);
// historyService.createHistoricVariableInstanceQuery().taskId(taskId);
// *****************************************************分隔符********************************************************************
// *****************************************************分隔符********************************************************************
// 可能还须要的API
// 挪动工作,人为跳转工作
// runtimeService.createChangeActivityStateBuilder().processInstanceId(processInstanceId)
// .moveActivityIdTo(currentActivityTaskId, newActivityTaskId).changeState();
// 如果在数据库配置了分组和用户,还会用到
List<User> users = identityService.createUserQuery().list(); //用户查问,用户id对应xml 外面配置的用户
List<Group> groups = identityService.createGroupQuery().list(); //分组查问,分组id对应xml 外面配置的分组 如 stu_group,te_group 在表里是id的值
// 另外,每个查问前面都能够拼条件,内置恁多查问,包含含糊查问,大小比拟都有
}
}
参考资料
【1】分享牛Flowable文档汉化:https://github.com/qiudaoke/f…
【2】猫七姑娘 flowable-6.6.0 运行官网 demo
【3】华格瑞沙 https://www.cnblogs.com/yangj…
版权申明:本文为CSDN博主「cy谭」的原创文章,遵循CC 4.0 BY-SA版权协定,转载请附上原文出处链接及本申明。\
链接:https://blog.csdn.net/zhan107…
近期热文举荐:
1.1,000+ 道 Java面试题及答案整顿(2022最新版)
2.劲爆!Java 协程要来了。。。
3.Spring Boot 2.x 教程,太全了!
4.别再写满屏的爆爆爆炸类了,试试装璜器模式,这才是优雅的形式!!
5.《Java开发手册(嵩山版)》最新公布,速速下载!
感觉不错,别忘了顺手点赞+转发哦!
发表回复