工作流之Flowable与SpringBoot结合

news/2024/11/15 17:35:22/文章来源:https://www.cnblogs.com/jingzh/p/18353066

目录
  • 1 Flowable
    • 1.1 flowable-ui部署运行
    • 1.2 绘制流程图
      • 1.2.1 绘制
      • 1.2.2 绘图细节
      • 1.2.3 bpmn文件导入
    • 1.3 后台项目搭建
      • 1.3.1 pom.xml
      • 1.3.2 数据库表说明
    • 1.4 流程引擎API与服务
      • 1.4.1 主要API
      • 1.4.2 示例

1 Flowable

1.1 flowable-ui部署运行

flowable-6.6.0 运行 官方demo
参考文档:https://flowable.com/open-source/docs/bpmn/ch14-Applications/

分享牛Flowable文档汉化:https://github.com/qiudaoke/flowable-userguide
华格瑞沙 https://www.cnblogs.com/yangjiming/p/10938515.html
部署步骤:

  • 从官网下载flowable-6.6.0 : https://github.com/flowable/flowable-engine/releases/download/flowable-6.6.0/flowable-6.6.0.zip

  • 将压缩包中的 flowable-6.6.0\wars\flowable-ui.war 丢到Tomcat中跑起来

  • 打开 http://localhost:8080/flowable-ui 用账户:admin/test 登录
    在这里插入图片描述

  • 进入 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,这样创建的流程后端程序就能直接使用

1.2 绘制流程图

1.2.1 绘制

在这里插入图片描述

根据业务需要在 flowable-ui -> APP.MODELER 里面绘制流程图,示例如上图。先解释一些概念。

  • 事件(event):通常用于为流程生命周期中发生的事情建模,图里是【开始、结束】两个圈。
  • 顺序流(sequence flow):是流程中两个元素间的连接器。图里是【箭头线段】。
  • 网关(gateway):用于控制执行的流向。图里是【菱形(中间有X)
  • 用户任务(user task):用于对需要人工执行的任务进行建模。图里是【矩形】。

简单的工作流大概就这些元素(还有很多这里就不扩展了)。下面描述一下工作流是如何流动的。

首先启动了工作流后,由【开始】节点自动流向【学生】节点,等待该任务执行。任务被分配的学生用户执行后流向 【老师】节点,再次等待该任务执行。被分配的老师用户执行后流向 【网关】,网关以此检查每个出口,流向符合条件的任务,比如这里老师执行任务时是同意,就流向【校长】节点,等待该任务执行。执行后跟老师类似,同意后就流向【结束】节点,整个流程到此结束。

1.2.2 绘图细节

保留流程模型
在这里插入图片描述

顺序流可以设置流条件来限制流动,比如上面的网关出口就设置了条件
在这里插入图片描述
任务需要分配任务的执行用户,可以分配到候选组,也可以直接分配到候选人
在这里插入图片描述

最后导出工作流文件
在这里插入图片描述

文件内容

<?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>

1.2.3 bpmn文件导入

如果需要,可以把这个流程文件下载下来,直接导入使用
在这里插入图片描述

1.3 后台项目搭建

后台项目基于jdk8,使用springboot框架

1.3.1 pom.xml

<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%2B8driver-class-name: com.mysql.jdbc.Driverusername: xxxxpassword: xxxx

1.3.2 数据库表说明

Flowable的所有数据库表都以ACT_开头。第二部分是说明表用途的两字符标示符。服务API的命名也大略符合这个规则。
Flowable建表机制:

  • 自动创建表:Flowable 在引擎配置中设置了数据库表的自动创建机制。只要引擎启动并连接到数据库,它会检查表是否存在。如果不存在,Flowable 会根据其内部的数据库脚本创建这些表。
  • 配置选项:可以在 Flowable 的配置文件中通过 databaseSchemaUpdate 属性来配置数据库表的创建和更新行为。
    • true 或 create-drop:引擎启动时创建表,停止时删除表(通常用于测试环境)。
    • false:不自动创建表,需要手动创建。
    • create:如果表不存在则创建。
    • drop-create:如果表存在则删除,然后创建。

各个表说明:

  • ACT_RE_RE代表repository。带有这个前缀的表包含静态信息,例如流程定义与流程资源(图片、规则等)。

  • ACT_RU_RU 代表runtime。这些表存储运行时信息,例如流程实例(process instance)、用户任务(user task)、变量(variable)、作业(job)等。Flowable只在流程实例运行中保存运行时数据,并在流程实例结束时删除记录。这样保证运行时表小和快。

  • ACT_HI_HI代表history。这些表存储历史数据,例如已完成的流程实例、变量、任务等。

  • ACT_GE_:通用数据。在多处使用。

  • 通用数据表(2个)
    act_ge_bytearray:二进制数据表,如流程定义、流程模板、流程图的字节流文件;
    act_ge_property:属性数据表(不常用);

  • 历史表(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:流程历史变量表,存储流程历史节点的变量信息;

  • 用户相关表(4个,IdentityService接口操作的表)
    act_id_group:用户组信息表,对应节点选定候选组信息;
    act_id_info:用户扩展信息表,存储用户扩展信息;
    act_id_membership:用户与用户组关系表;
    act_id_user:用户信息表,对应节点选定办理人或候选人信息;

  • 流程定义、流程模板相关表(3个,RepositoryService接口操作的表)
    act_re_deployment:部属信息表,存储流程定义、模板部署信息;
    act_re_procdef:流程定义信息表,存储流程定义相关描述信息,但其真正内容存储在act_ge_bytearray表中,以字节形式存储;
    act_re_model:流程模板信息表,存储流程模板相关描述信息,但其真正内容存储在act_ge_bytearray表中,以字节形式存储;

  • 流程运行时表(6个,RuntimeService接口操作的表)
    act_ru_task:运行时流程任务节点表,存储运行中流程的任务节点信息,重要,常用于查询人员或部门的待办任务时使用;
    act_ru_event_subscr:监听信息表,不常用;
    act_ru_execution:运行时流程执行实例表,记录运行中流程运行的各个分支信息(当没有子流程时,其数据与act_ru_task表数据是一一对应的);
    act_ru_identitylink:运行时流程人员表,重要,常用于查询人员或部门的待办任务时使用;
    act_ru_job:运行时定时任务数据表,存储流程的定时任务信息;
    act_ru_variable:运行时流程变量数据表,存储运行中的流程各节点的变量信息;

1.4 流程引擎API与服务

1.4.1 主要API

引擎API是与Flowable交互的最常用手段。总入口点是ProcessEngine。
在这里插入图片描述
说明:

  • RepositoryService:很可能是使用 Flowable引擎要用的第一个服务。这个服务提供了管理与控制部署(deployments)与流程定义(process definitions)的操作。管理静态信息,
  • RuntimeService:用于启动流程定义的新流程实例。
  • IdentityService:很简单,它用于管理(创建,更新,删除,查询……)组与用户。
  • FormService:是可选服务。也就是说Flowable没有它也能很好地运行,而不必牺牲任何功能。
  • HistoryService:暴露Flowable引擎收集的所有历史数据。要提供查询历史数据的能力。
  • ManagementService:通常在用Flowable编写用户应用时不需要使用。它可以读取数据库表与表原始数据的信息,也提供了对作业(job)的查询与管理操作。
  • DynamicBpmnService:可用于修改流程定义中的部分内容,而不需要重新部署它。例如可以修改流程定义中一个用户任务的办理人设置,或者修改一个服务任务中的类名。

1.4.2 示例

接下来使用之前的请假流程图,上代码

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*/
@Slf4j
public class TestFlowable {@Autowiredprivate RepositoryService repositoryService;@Autowiredprivate RuntimeService runtimeService;@Autowiredprivate HistoryService historyService;@Autowiredprivate org.flowable.engine.TaskService taskService;@Autowiredprivate org.flowable.engine.IdentityService identityService;public void createDeploymentZip() {/** 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();}/** Step 2: 查询部署的流程定义*/List<ProcessDefinition> list = repositoryService.createProcessDefinitionQuery().processDefinitionKey("leave_approval").list();List<ProcessDefinition> pages = repositoryService.createProcessDefinitionQuery().processDefinitionKey("leave_approval").listPage(1, 30);/** 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());/** 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); //删除实例/** 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());}/** 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);}/** 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的值// 另外,每个查询后面都可以拼条件,内置恁多查询,包括模糊查询,大小比较都有}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/781142.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Docker简单使用MySQL

前提 安装好Docker 下载Mysql镜像 在这个网址搜索Mysql并选择合适的镜像 https://hub.atomgit.com/详情页就有拉取镜像的命令查看Docker所有镜像 ​ docker images镜像重命名 ​ docker tag 镜像id 仓库:标签 ​ docker tag 8a709252ac32 mysql:5.7 镜像启动(容器) docker ru…

上云避坑指南

云计算的本质就俩词:共享、服务。上云的目的是为了让自己聚焦于业务我在之前的文章《云计算-虚拟化-OpenStack》里聊过,云计算的本质是一种IT资源通过虚拟化进行的共享,是一种更高维度的服务。云计算的本质就俩词:共享、服务。 1、建议企业上云 作为一个IT行业14余年的老杆…

简单在 WinUI 仿造 WPF 的 ColumnDefinition SharedSizeGroup 共享列宽功能

本文将告诉大家如何在 WinUI 3 或 UNO 里面,仿造 WPF 的 ColumnDefinition SharedSizeGroup 共享列宽功能本文的实现代码是大量从 https://github.com/Qiu233/WinUISharedSizeGroup 抄的,感谢大佬提供的代码。我在此基础上简化了对 Behavior 的依赖,在本文末尾放上了全部代码…

ext2 文件系统解析

ext2文件系统整体布局 每个块组内部都有相关的元数据对该块组进行管理。如图所示,第一个块组中的元数据包括引导块、超级块、块组描述符、预留GDT块、数据块位图、inode位图、inode表和其它数据块。后续块组中有些是对超级块的备份,有些则没有第一个块组这么完整的元数据信息…

mac最新安装php各版本教程,和ghcr.io被墙的解决方法

2024年8月,很多国外链接都被墙了,导致mac安装php旧版本好困难,特意记录一下方法 1.brew tap shivammathur/php 超时问题如果 tap 超时, 则用浏览器打开 https://github.com/shivammathur/homebrew-php , 下载压缩包 或者直接百度云下载 百度云:https://pan.baidu.com/s/1lsp-E…

澳洲 WHV All In One

澳洲 WHV All In One 工作假期签证申请常见问题 Cambly 英语外教,真人视频口语对话练习课澳洲 WHV All In One WHVWorking Holiday VisaFor young adults who want an extended holiday and to work here to fund it.https://immi.homeaffairs.gov.au/visas/getting-a-visa/vi…

024.Vue3入门,父页面给子页面传递多种数据

1、App.vue代码如下:<template><Father/> </template><script setup> import Father from ./view/Father.vue </script><style> </style>2、Father.vue代码如下:<template><h3>父页面</h3><Child :FMsg=&quo…

小总结(1)

前言: 这篇总结本来想在学习reverse一周年的时候写的,回过头想想,我的大学生活不只有reverse,每一个成长的瞬间都应该被记录下来。 Happiness should be about everything and not a certain moment (至于为什么选择使用博客园写,我的GitHub使用无法将本地资源上传,导致…

023.Vue3入门,父页面给子页面传递数据

1、App.vue代码如下:<template><Father/> </template><script setup> import Father from ./view/Father.vue </script><style> </style>2、Father.vue代码如下:<template><h3>父页面</h3><Child :title=&qu…

022.Vue3入门,注册全局组件,在任何页面直接使用

1、main.js代码如下:// import ./assets/main.cssimport {createApp} from vue import App from ./App.vue import Config from "@/config.js"; import Testpage001 from "@/view/Testpage001.vue";const app = createApp(App);// 定义一个全局组件,名字为…

021.Vue3入门,注册全局组件,在任何页面直接使用

1、main.js代码如下:// import ./assets/main.cssimport {createApp} from vue import App from ./App.vue import Config from "@/config.js"; import Testpage001 from "@/view/Testpage001.vue";const app = createApp(App);// 定义一个全局组件,名字为…

支付三大黑盒之三账务核心

各位小伙伴大家好! 这次给大家揭秘支付三大黑盒的最后一个“账务核心”(另外两个是清结算对账、支付引擎),这账务核心可能是其中门槛最高的,因为他既要懂会计知识,又有懂技术如何实现高性能的记账。 下面我就用大白话+图片的方式给大家来详细介绍支付系统最后一个黑盒“账…