Spring Boot集成activiti快速入门Demo

1.什么事activiti?

Activiti是一个工作流引擎,可以将业务系统中复杂的业务流程抽取出来,使用专门的建模语言BPMN2.0进行定义,业务流程按照预先定义的流程进行执行,实现了系统的流程流activiti进行管理,减少业务系统由于流程变更进行系统升级改造的工作量,从而提高系统的健壮性,同时也减少了系统开发维护成本.

Activiti 的核心服务组件

  1. RepositoryService:提供一系列管理流程部署和流程定义的API。
  2. RuntimeService:在流程运行时对流程实例进行管理与控制。
  3. TaskService:对流程任务进行管理,例如任务提醒、任务完成和创建任务等。
  4. IdentityService:提供对流程角色数据进行管理的API,这些角色数据包括用户组、用户及它们之间的关系。
  5. ManagementService:提供对流程引擎进行管理和维护的服务。
  6. HistoryService:对流程的历史数据进行操作,包括查询、删除这些历史数据。
  7. FormService:表单服务。

业务流程模型 BPMN xml 配置文件

一个xml文件,activiti去解析这个文件,了解我们到底想干什么事。可以使用ideal插件【Activiti BPMN visualizer】来编辑这个文件,插件具体使用方法可以Google搜索一下,这里就不具体描述了

2

2.mysql数据库环境搭建

为什么需要mysql?

因为activiti会生成很多流程执行的表。表都是 ACT 开头,用以记录工作流产生的记录数据!如下图所示

表结构介绍

表分类表名解释
一般数据
ACT_GE_BYTEARRAY通用的流程定义和流程资源
ACT_GE_PROPERTY系统相关属性
流程历史记录
ACT_HI_ACTINST历史的流程实例
ACT_HI_ATACHMENT历史的流程附件
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_IDNTITYLINK运行时用户关系信息,存储任务节点与参与者的相关信息
ACT_RU_JOB运行时作业
ACT_RU_TASK运行时任务
ACT_RU_VARIABLE运行时变量表

安装

docker run --name docker-mysql-5.7 -e MYSQL_ROOT_PASSWORD=123456 -p 3306:3306 -d mysql:5.7

初始化

create database activity;

默认用户和密码

msyql username:root
mysql password:123456

3.代码工程

实现目的:实现流程驱动

pom.xml

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"><parent><artifactId>springboot-demo</artifactId><groupId>com.et</groupId><version>1.0-SNAPSHOT</version></parent><modelVersion>4.0.0</modelVersion><artifactId>activiti</artifactId><properties><maven.compiler.source>8</maven.compiler.source><maven.compiler.target>8</maven.compiler.target></properties><dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-autoconfigure</artifactId></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-test</artifactId><scope>test</scope></dependency><dependency><groupId>org.activiti</groupId><artifactId>activiti-spring-boot-starter-basic</artifactId><version>6.0.0</version></dependency><dependency><groupId>mysql</groupId><artifactId>mysql-connector-java</artifactId><scope>runtime</scope></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-jpa</artifactId></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-security</artifactId></dependency></dependencies>
</project>

application.properties

spring.datasource.driver-class-name=com.mysql.jdbc.Driver
spring.datasource.url=jdbc:mysql://127.0.0.1:3306/activity?characterEncoding=utf8&useSSL=true
spring.datasource.username=root
spring.datasource.password=123456
spring.jpa.properties.hibernate.hbm2ddl.auto=update
spring.jpa.show-sql=true
server.session.timeout=10
server.tomcat.uri-encoding=UTF-8
server.port=8088

test.bpmn20.xml

<?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-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:activiti="http://activiti.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" xmlns:dc="http://www.omg.org/spec/DD/20100524/DC" xmlns:di="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.activiti.org/testm1510735932336" id="m1510735932336" name=""><process id="leave" isExecutable="true" isClosed="false" processType="None"><startEvent id="_2" name="StartEvent"/><endEvent id="_3" name="EndEvent"/><userTask id="approve" name="manager approve" activiti:assignee="${approve}"/><exclusiveGateway id="_5" name="ExclusiveGateway"/><sequenceFlow id="_6" sourceRef="approve" targetRef="_5"/><sequenceFlow id="_7" name="pass" sourceRef="_5" targetRef="_3"><conditionExpression xsi:type="tFormalExpression"><![CDATA[${pass}]]></conditionExpression></sequenceFlow><userTask id="application" name="apply submit" activiti:assignee="${apply}"/><sequenceFlow id="_9" sourceRef="_2" targetRef="application"/><sequenceFlow id="_10" sourceRef="application" targetRef="approve"/><userTask id="modify" name="modify apply" activiti:assignee="${apply}"/><sequenceFlow id="_12" name="fail " sourceRef="_5" targetRef="modify"><conditionExpression xsi:type="tFormalExpression"><![CDATA[${!pass}]]></conditionExpression></sequenceFlow><sequenceFlow id="_13" sourceRef="modify" targetRef="approve"/></process><bpmndi:BPMNDiagram id="BPMNDiagram_leave"><bpmndi:BPMNPlane bpmnElement="leave" id="BPMNPlane_leave"><bpmndi:BPMNShape bpmnElement="_2" id="BPMNShape__2"><omgdc:Bounds height="35.0" width="35.0" x="15.0" y="60.0"/></bpmndi:BPMNShape><bpmndi:BPMNShape bpmnElement="_3" id="BPMNShape__3"><omgdc:Bounds height="35.0" width="35.0" x="630.0" y="63.0"/></bpmndi:BPMNShape><bpmndi:BPMNShape bpmnElement="approve" id="BPMNShape_approve"><omgdc:Bounds height="55.0" width="85.0" x="315.0" y="50.0"/></bpmndi:BPMNShape><bpmndi:BPMNShape bpmnElement="_5" id="BPMNShape__5"><omgdc:Bounds height="40.0" width="40.0" x="505.0" y="60.5"/></bpmndi:BPMNShape><bpmndi:BPMNShape bpmnElement="application" id="BPMNShape_application"><omgdc:Bounds height="55.0" width="85.0" x="135.0" y="50.0"/></bpmndi:BPMNShape><bpmndi:BPMNShape bpmnElement="modify" id="BPMNShape_modify"><omgdc:Bounds height="55.0" width="85.0" x="315.0" y="150.0"/></bpmndi:BPMNShape><bpmndi:BPMNEdge bpmnElement="_6" id="BPMNEdge__6"><omgdi:waypoint x="400.0" y="77.0"/><omgdi:waypoint x="505.0" y="80.5"/></bpmndi:BPMNEdge><bpmndi:BPMNEdge bpmnElement="_7" id="BPMNEdge__7"><omgdi:waypoint x="545.0" y="80.5"/><omgdi:waypoint x="630.0" y="80.0"/></bpmndi:BPMNEdge><bpmndi:BPMNEdge bpmnElement="_9" id="BPMNEdge__9"><omgdi:waypoint x="50.0" y="77.0"/><omgdi:waypoint x="135.0" y="77.0"/></bpmndi:BPMNEdge><bpmndi:BPMNEdge bpmnElement="_10" id="BPMNEdge__10"><omgdi:waypoint x="220.0" y="77.0"/><omgdi:waypoint x="315.0" y="77.0"/></bpmndi:BPMNEdge><bpmndi:BPMNEdge bpmnElement="_12" id="BPMNEdge__12"><omgdi:waypoint x="525.0" y="100.5"/><omgdi:waypoint x="525.0" y="177.0"/><omgdi:waypoint x="400.0" y="177.0"/></bpmndi:BPMNEdge><bpmndi:BPMNEdge bpmnElement="_13" id="BPMNEdge__13"><omgdi:waypoint x="357.0" y="150.0"/><omgdi:waypoint x="357.0" y="105.0"/></bpmndi:BPMNEdge></bpmndi:BPMNPlane></bpmndi:BPMNDiagram>
</definitions>

controller

package com.et.activiti.controller;import com.et.activiti.service.ActivityConsumerService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;import java.util.HashMap;
import java.util.Map;@RestController
public class HelloWorldController {@AutowiredActivityConsumerService activityConsumerService;@RequestMapping(value="/startActivityDemo",method= RequestMethod.GET)public boolean startActivityDemo(){return activityConsumerService.startActivityDemo();}
}

service

package com.et.activiti.service;import org.activiti.engine.task.TaskQuery;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;public interface ActivityConsumerService {public boolean startActivityDemo();}
package com.et.activiti.service;import java.util.HashMap;
import java.util.Map;import org.activiti.engine.RuntimeService;
import org.activiti.engine.TaskService;
import org.activiti.engine.impl.persistence.entity.ExecutionEntity;
import org.activiti.engine.task.Task;
import org.activiti.engine.task.TaskQuery;
import org.apache.commons.lang3.StringUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;@Service
public class ActivityConsumerServiceImpl implements ActivityConsumerService {@Autowiredprivate RuntimeService runtimeService;@Autowiredprivate TaskService taskService;@Overridepublic boolean startActivityDemo() {System.out.println("method startActivityDemo begin....");Map<String, Object> map = new HashMap<String, Object>();map.put("apply", "zhangsan");map.put("approve", "lisi");//flow startExecutionEntity pi1 = (ExecutionEntity) runtimeService.startProcessInstanceByKey("leave", map);String processId = pi1.getId();pi1.getExecutions().forEach(row->{String taskId =  row.getTasks().get(0).getId();taskService.complete(taskId, map);//complete first stepTask task = taskService.createTaskQuery().processInstanceId(processId).singleResult();String taskId2 = task.getId();map.put("pass", false);taskService.complete(taskId2, map);//refuse applySystem.out.println("method startActivityDemo end....");});return false;}
}

以上只是一些关键代码,所有代码请参见下面代码仓库

代码仓库

  • GitHub - Harries/springboot-demo: a simple springboot demo with some components for example: redis,solr,rockmq and so on.

4.测试

  • 启动Spring Boot应用
  • 访问http://127.0.0.1:8088/startActivityDemo

可能需要你先登录一下,因为activiti默认集成Spring security,默认的用户名user,密码是程序启动会自动生成一个,如下所示

Using generated security password: 86f027ef-93c2-46ee-a054-c1ada840e64d

5.引用参考

  • Spring Boot集成activiti快速入门Demo | Harries Blog™
  • Activiti User Guide

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

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

相关文章

做抖店的门槛高吗?一个月的时间能入门吗?基础问题解答如下

我是王路飞。 抖店&#xff0c;依旧是普通人做抖音最好的渠道&#xff0c;没有之一&#xff0c;依旧值得我们all in。 这是我对2024年抖音小店的看法和态度&#xff0c; 那么做抖店的门槛高吗&#xff1f;新手用一个月的时间能做到入门吗&#xff1f;投入和回报的数据是多少…

OpenAI 推出革命性新模型 GPT-4o:全能AI的新纪元

GPT-4o 模型的推出预示着人工智能领域的又一次飞跃&#xff0c;它将如何改变我们的世界&#xff1f; 在人工智能的快速发展浪潮中&#xff0c;OpenAI 再次站在了技术革新的前沿。2024年5月14日&#xff0c;OpenAI 宣布了其最新旗舰模型 GPT-4o&#xff0c;这不仅是一个简单的版…

一文讀懂DNS代理伺服器

DNS&#xff08;功能變數名稱系統&#xff09;作為互聯網的基礎組件之一&#xff0c;它將可讀的功能變數名稱&#xff08;如www.example.com&#xff09;轉換為電腦能夠識別的IP地址&#xff08;如192.0.2.1&#xff09;。每打開一個網頁&#xff0c;查看郵件&#xff0c;甚至線…

【C++小语法技巧】命名空间和输入输出

在使用C语言编程过程中&#xff0c;C语言的要求之严格&#xff0c;编程过程之繁琐&#xff0c;大同小异的重复性工作&#xff0c;令C之父使用C语言编程时也深受其扰&#xff0c;于是乎C兼容C小语法诞生了 一、命名空间域&#xff08;解决C语言中命名冲突&#xff09; 1.定义命…

Python—基本数据类型

Python是一种解释型语言&#xff0c;非常灵活&#xff0c;并且具有很高的交互性&#xff0c;可以方便的编写、测试和调试代码&#xff1b;本文主要介绍Python的基本数据类型&#xff0c;主要包括整数、浮点数、字符串、布尔、列表、元组、集合和字典。 目前大部分是用Python进…

C语言之旅:文件操作

目录 一什么是文件 1.1程序文件&#xff1a; 1.2数据文件&#xff1a; 1.3文件名 二.文件的打开与关闭 2.1流和标准流 2.2 文件指针 2.3文件的打开与关闭 三.文件的顺序读写 3.1顺序读写函数介绍 fgetc fputc fgets fputs 四.文件的随机读写 4.1 fseek 4.2 ftell…

【案例教程】土地利用/土地覆盖遥感解译与基于CLUE模型未来变化情景预测

查看原文>>>土地利用/土地覆盖遥感解译与基于CLUE模型未来变化情景预测 土地利用/土地覆盖数据是生态、环境和气象等领域众多模型的重要输入参数之一。基于遥感影像解译&#xff0c;可获取历史或当前任何一个区域的土地利用/土地覆盖数据&#xff0c;用于评估区域的生…

FISCO BCOS助力深港跨境数据验证平台上线试运行

日前&#xff0c;深港跨境数据验证平台在深港两地上线试运行。这是在《粤港澳大湾区发展规划纲要》和《全面深化前海深港现代服务业合作区改革开放方案》等政策指引下&#xff0c;深港两地加快发展新质生产力、扎实推进数字经济高质量发展的又一创新实践。 深港跨境数据验证平…

Rust语言内部运行原理介绍

Rust语言因其独特的内存管理特性和安全性而日益受到程序员的青睐。本文旨在为初学者提供一个关于Rust的基础入门课程和对其内部运行原理的简介&#xff0c;帮助你理解这门语言的核心优势和设计理念。 1. Rust入门课程 基础知识 Rust语言的学习起点应从其基本语法开始。Rust的…

I2C总线协议

目录 一、前言 二、概念 三、I2C总线 1、主从结构 2、两线 3、半双工 4、上拉电阻 5、高阻态 四、时序图 1、起止信号 2、数据传输 3、应答信号 五、数据传送 1、传送格式 2、读写数据 六、时钟同步与仲裁 1、时钟同步 2、仲裁 七、时钟拉伸 八、总线速率 一、…

Kotlin核心编程知识点-01-基础语法

文章目录 0.前言1.不一样的类型声明1.1.增强的类型推导1.2.声明函数返回值类型1.3.是否需要显示声明类型&#xff1f; 2.val和var的使用规则2.1.val的含义&#xff1a;引用不可变 3.高阶函数和Lambda3.1.抽象和高阶函数3.2.函数作为参数的需求3.3.函数的类型3.4.方法和成员引用…

项目实施奖励方式调查

此调查主要了解各企业的项目激励方式&#xff08;只有3个问题&#xff09;。 相应的调查结果也将定时公布&#xff0c;希望对企业项目管理、企业激励方面有一定帮助。 感谢大家的支持。 链接地址为&#xff1a;项目实施奖励方式调查

数据的可计算性

目的&#xff1a;更容易被计算机程序分析和处理。所以涉及到数据准备、清洗、整理、格式化等过程。 1、标准化数据格式&#xff1a;所有数据遵循统一的格式和标准。比如日期和时间以标准格式存储。 2、清洗数据&#xff1a;移除或修正 错误数据和异常值&#xff0c;包括去除重…

Flutter 依据JSON数据自动生成实体类

json自动化生成工具 点击这里可以跳转 页面是这样的 然后在左边输入你的json数据&#xff0c;它会自动生成对应的实体类 生成的实体类是如下&#xff1a; import package:json_annotation/json_annotation.dart; part merch_region.g.dart;JsonSerializable()class MerchReg…

国网645协议报文解析软件

本文分享一款支持国网DL645-2007规约的报文解析软件&#xff0c; 链接: https://pan.baidu.com/s/1ngbBG-yL8ucRWLDflqzEnQ 提取码: y1de 主界面如下图所示&#xff1a; 本解析软件同时内嵌规约文档&#xff0c;支持一键打开文档&#xff0c;功能如下&#xff1a; 同时支持模…

堆的数组实现

前言 本次博客来讲解一下堆的数组实现&#xff0c;好吧还是会结合图例&#xff0c;让大家理解 堆的定义 什么是堆&#xff1f; 堆是一颗完全二叉树。它的性质是父节点一定大于或者一定小于子节点 每一个结点都要满足这个性质就是堆 堆的特性是堆顶的数据一定是最大或最小…

ROS仿真多点导航

仿真环境启动&#xff1a; 1、启动并进入到相应环境&#xff1a; roscarroscar-virtual-machine:~/artcar_simulation$ 启动gazebo环境&#xff1a; roslaunch artcar_gazebo artcar_gazebo.launch 启动move_base&#xff1a; roslaunch artcar_nav artcar_move_base.launc…

3D数字化技术如何改变汽车行业?

近年来&#xff0c;新能源汽车行业加速发展&#xff0c;新车型密集发布&#xff0c;汽车保有量和车龄的增加&#xff0c;也同时点燃了汽车后市场的增长引擎。对于车企而言&#xff0c;如何全方面优化汽车从研发、生产、售后到营销的各个环节&#xff0c;以便适应快速变化的市场…

中文域名和英文域名有什么区别?中文域名有哪些优势?

作为企业网站的重要标识&#xff0c;域名在数字化时代扮演着重要角色。而随着中国互联网的快速发展&#xff0c;中文域名也逐渐崭露头角&#xff0c;受到国内企业和用户的关注。本文国科云作为中国科学院控股有限公司旗下的域名管理品牌&#xff0c;简单为大家介绍下中文域名和…

使用make_blobs生成数据并使用KNN机器学习算法进行分类和预测以及可视化

生成数据 使用make_blobs生成数据并使用matplotlib进行可视化 完整代码&#xff1a; from sklearn.datasets import make_blobs # KNN 分类器 from sklearn.neighbors import KNeighborsClassifier # 画图工具 import matplotlib.pyplot as plt # 数据集拆分工具 from sklea…