spring-statemachine 状态机自定义持久化入库

使用 spring-statemachine 状态机持久化时,可以通过内存、spring-statemachine-redis 或 spring-statemachine-data-jpa 现有方式持久化处理。

因项目审核操作记录频繁,数据量大,使用 内存 或 spring-statemachine-redis 模式不可取,而项目使用的是 MyBatis,使用 spring-statemachine-data-jpa 也不合适,需要自定义实现,简述步骤如下:

一、引入依赖

            <!--状态机--><dependency><groupId>org.springframework.statemachine</groupId><artifactId>spring-statemachine-starter</artifactId><version>2.2.3.RELEASE</version></dependency><dependency><groupId>org.springframework.statemachine</groupId><artifactId>spring-statemachine-kryo</artifactId><version>1.2.14.RELEASE</version></dependency>

二、创建持久化记录存储表

CREATE TABLE `state_machine_context` (`id` BIGINT UNSIGNED NOT NULL AUTO_INCREMENT COMMENT '自增主键',`machine_type` VARCHAR (32) DEFAULT '' COMMENT '状态机类型',`machine_id` VARCHAR (36) DEFAULT '' COMMENT '状态机ID',`machine_data` TINYBLOB COMMENT '状态机数据',`machine_state` VARCHAR (32) DEFAULT '' COMMENT '状态机状态',`machine_event` VARCHAR (36) DEFAULT '' COMMENT '状态机事件',`create_date` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',`update_date` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '更新时间',PRIMARY KEY (`id`),KEY `idx_machine_id` (`machine_id`)
) ENGINE = INNODB COMMENT = '状态机上下文'

关键字段说明 

  • machine_type:标记业务类型,如订单业务、用户业务
  • macheine_id:业务数据ID,如订单ID、用户ID
  • machine_data:状态机二进制数据

其它字段可根据自己业务需求自定义

三、自定义持久化类,即实现接口 StateMachinePersist

import com.esotericsoftware.kryo.Kryo;
import com.esotericsoftware.kryo.io.Input;
import com.esotericsoftware.kryo.io.Output;
import org.apache.commons.lang3.tuple.Pair;
import org.springframework.messaging.MessageHeaders;
import org.springframework.statemachine.StateMachineContext;
import org.springframework.statemachine.StateMachinePersist;
import org.springframework.statemachine.kryo.MessageHeadersSerializer;
import org.springframework.statemachine.kryo.StateMachineContextSerializer;
import org.springframework.statemachine.kryo.UUIDSerializer;import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.util.Date;
import java.util.Objects;
import java.util.Optional;
import java.util.UUID;/*** @author songjianyong*/
public class CustomStateMachinePersist<S, E> implements StateMachinePersist<S, E, Pair<String, String>> {private static final ThreadLocal<Kryo> KRYO_THREAD_LOCAL = ThreadLocal.withInitial(() -> {Kryo kryo = new Kryo();kryo.addDefaultSerializer(StateMachineContext.class, new StateMachineContextSerializer());kryo.addDefaultSerializer(MessageHeaders.class, new MessageHeadersSerializer());kryo.addDefaultSerializer(UUID.class, new UUIDSerializer());return kryo;});private byte[] serialize(StateMachineContext<S, E> context) {Kryo kryo = KRYO_THREAD_LOCAL.get();ByteArrayOutputStream out = new ByteArrayOutputStream();Output output = new Output(out);kryo.writeObject(output, context);output.close();return out.toByteArray();}@SuppressWarnings("unchecked")private StateMachineContext<S, E> deserialize(byte[] data) {if (data == null || data.length == 0) {return null;}Kryo kryo = KRYO_THREAD_LOCAL.get();ByteArrayInputStream in = new ByteArrayInputStream(data);Input input = new Input(in);return kryo.readObject(input, StateMachineContext.class);}private final StateMachineContextDao stateMachineContextDao;public SongStateMachinePersist(StateMachineContextDao stateMachineContextDao) {this.stateMachineContextDao = stateMachineContextDao;}@Overridepublic void write(StateMachineContext<S, E> context, Pair<String, String> pair) throws Exception {byte[] machineData = serialize(context);String machineId = pair.getKey();String machineType = pair.getValue();StateMachineContextEntity stateMachineContexEntity = stateMachinePersistDao.findByMachineIdAndMachineType(machineId, machineType);if (Objects.nonNull(stateMachineContexEntity)) {stateMachineContexEntity.setMachineData(machineData);stateMachineContexEntity.setMachineState(Optional.ofNullable(context.getState()).map(Objects::toString).orElse(stateMachineContexEntity.getMachineState()));stateMachineContexEntity.setMachineEvent(Optional.ofNullable(context.getEvent()).map(Objects::toString).orElse(stateMachineContexEntity.getMachineEvent()));stateMachineContexEntity.setUpdateDate(new Date());stateMachineContextDao.updateById(stateMachineContexEntity);return;}StateMachineContextEntity entity = new StateMachineContextEntity();entity.setMachineId(machineId);entity.setMachineData(machineData);entity.setMachineType(machineType);entity.setMachineState(Optional.ofNullable(context.getState()).map(Objects::toString).orElse(null));entity.setMachineEvent(Optional.ofNullable(context.getEvent()).map(Objects::toString).orElse(null));stateMachineContextDao.save(entity);}@Overridepublic StateMachineContext<S, E> read(Pair<String, String> pair) throws Exception {String machineId = pair.getKey();String machineType = pair.getValue();StateMachineContextEntity stateMachineContexEntity = stateMachineContextDao.findByMachineIdAndMachineType(machineId, machineType);if (Objects.isNull(stateMachineContexEntity)) {return null;}byte[] machineData = stateMachineContexEntity.getMachineData();return deserialize(machineData);}
}

四、自定义状态机,即继承类 AbstractStateMachinePersister

import org.apache.commons.lang3.tuple.Pair;
import org.springframework.statemachine.StateMachinePersist;
import org.springframework.statemachine.persist.AbstractStateMachinePersister;/*** @author songjianyong*/
public class CustomStateMachinePersister<S, E> extends AbstractStateMachinePersister<S, E, Pair<String, String>> {public CustomStateMachinePersister(StateMachinePersist<S, E, Pair<String, String>> stateMachinePersist) {super(stateMachinePersist);}
}

五、使用自定义状态机

    /*** 持久化到库中** @return 数据库持久化状态机*/@Bean(name = "customStateMachinePersister")public CustomStateMachinePersister<S, E> customStateMachinePersister(StateMachineContextDao stateMachineContextDao) {CustomStateMachinePersist<S, E> customStateMachinePersist = new CustomStateMachinePersist<>(stateMachineContextDao);return new CustomStateMachinePersister<>(customStateMachinePersist);}
    @Resourceprivate StateMachinePersister<OrderStatus, OrderStatusChangeEvent, Pair<String, String> pair> customStateMachinePersister;

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

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

相关文章

QTday02(常用类、UI界面下的开发、信号与槽)

今日任务 1. 使用手动连接&#xff0c;将登录框中的取消按钮使用qt4版本的连接到自定义的槽函数中&#xff0c;在自定义的槽函数中调用关闭函数 将登录按钮使用qt5版本的连接到自定义的槽函数中&#xff0c;在槽函数中判断ui界面上输入的账号是否为"admin"&#x…

v-model表单数据双向绑定-表单提交示例

示例如下&#xff1a; <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>v-model表单数据双向绑定<…

354 俄罗斯套娃信封问题(贪心+二分)

题目 链接 给你一个二维整数数组 envelopes &#xff0c;其中 envelopes[i] [wi, hi] &#xff0c;表示第 i 个信封的宽度和高度。 当另一个信封的宽度和高度都比这个信封大的时候&#xff0c;这个信封就可以放进另一个信封里&#xff0c;如同俄罗斯套娃一样。 请计算 最多…

【17.19消失的两个数字】

目录 一、题目描述二、算法原理三、代码实现 一、题目描述 二、算法原理 三、代码实现 class Solution { public:vector<int> missingTwo(vector<int>& nums) {//找到a^bint temp0;for(auto num:nums){temp^num;}for(int i1;i<nums.size()2;i){temp^i;}//找…

Flink测试利器之DataGen初探 | 京东云技术团队

什么是 Flinksql Flink SQL 是基于 Apache Calcite 的 SQL 解析器和优化器构建的&#xff0c;支持ANSI SQL 标准&#xff0c;允许使用标准的 SQL 语句来处理流式和批处理数据。通过 Flink SQL&#xff0c;可以以声明式的方式描述数据处理逻辑&#xff0c;而无需编写显式的代码…

React 路由总结 react-router-dom6+react-router-dom5

开题 单页面应用和多页面应用 SPA&#xff1a;单页面应用程序&#xff0c;整个应用中只有一个页面(index.html) MPA&#xff1a;多页面应用程序&#xff0c;整个应用中有很多页面(*.html) react路由 现在的前端应用大多都是SPA单页面应用程序&#xff0c;也就是一个HTML页面的…

百度OCR识别图片文本字符串——物联网上位机软件

一、开发背景 根据项目需求&#xff0c;我们需要完成LED显示屏实时显示歌词的效果。最优的方法是调用歌曲播放器的API获取歌词&#xff0c;但是由于这个开发资格不是很好申请&#xff0c;因此我们采用其他方案&#xff0c;即通过OCR识别获取歌词&#xff0c;并投射到LED显示屏上…

二、电脑装机实践

计算机装机实践 一、零部件组装与启动二、操作系统安装与使用2.0 先导内容&#xff1a;BIOS引导模式、Boot Loader-引导加载程序、硬盘分区模式2.1 Win11安装2.2 Ubuntu22.042.2.0 Ubuntu简介2.2.1 Ubuntu22.04安装2.2.2 Ubuntu22.04使用2.2.2.1 新建文件模版2.2.2.2 给新安装应…

ExposureDiffusion: Learning to Expose for Low-light Image Enhancement论文阅读笔记

南洋理工大学、鹏城实验室、香港理工大学在ICCV2023发表的暗图增强论文。用diffusion模型来进行raw图像暗图增强&#xff0c;同时提出了一个自适应的残差层用来对具有不同信噪比的不同区域采取不同的去噪策略。 方法的框图如下所示&#xff1a; 一张raw图片可以由信号和噪声…

如何解决idea运行出现java: 程序包XX不存在

原因一&#xff1a;pom.xml没有将程序包XXX的<dependency>配置 直接将程序包的<dependency>配置写上即可 原因二&#xff1a;maven配置问题 打开idea的Settings, 进入Build Execution Deployment/Build Tools/Maven/Runner,将Delegate IDE勾上就可以了。(将idea…

Maven 基础教程系列

Maven是一个项目开发管理和理解工具。基于项目对象模型的概念&#xff1a;构建、依赖关系管理、文档创建、站点发布和分发发布都由pom.xml声明性文件控制。Maven可以通过插件进行扩展&#xff0c;以使用许多其他开发工具来报告或构建过程。 一、Maven 使用教程-CSDN博客 二、…

Eclipse Xtext 实现PLC ST 语言到C的转换

Eclipse Xtext 是开发领域专用语言&#xff08;DSL&#xff09;的工具。例如数据库的SQL 语言&#xff0c;PLC 的ST 语言都是一种领域专用语言。在开放自动化领域&#xff0c;提倡基于模型的设计方法。DSL 是描述模型的强有力工具。 在开发PLC 程序IDE时&#xff0c;开发ST编译…

网络安全内网渗透之信息收集--systeminfo查看电脑有无加域

systeminfo输出的内容很多&#xff0c;包括主机名、OS名称、OS版本、域信息、打的补丁程序等。 其中&#xff0c;查看电脑有无加域可以快速搜索&#xff1a; systeminfo|findstr "域:" 输出结果为WORKGROUP&#xff0c;可见该机器没有加域&#xff1a; systeminfo…

LeetCode【17】电话号码的字母组合

题目&#xff1a; 思路&#xff1a; 参考&#xff1a;https://blog.csdn.net/weixin_46429290/article/details/121888154 和上一个题《子集》的思路一样&#xff0c;先画出树结构&#xff0c;看树的深度&#xff08;遍历层级&#xff09;&#xff0c;树的宽度&#xff08;横向…

10.18作业

使用手动连接&#xff0c;将登录框中的取消按钮使用qt4版本的连接到自定义的槽函数中&#xff0c;在自定义的槽函数中调用关闭函数 将登录按钮使用qt5版本的连接到自定义的槽函数中&#xff0c;在槽函数中判断ui界面上输入的账号是否为"admin"&#xff0c;密码是否为…

IntelliJ IDEA中使用“Generate equals() and hashCode()“提供不同模板的区别

当在IntelliJ IDEA中使用"Generate equals() and hashCode()"功能时&#xff0c;它提供了几个模板选项来生成equals()和hashCode()方法的代码。以下是这些模板的一些区别&#xff1a; IntelliJ Default&#xff08;IntelliJ 默认&#xff09;&#xff1a;这是Intelli…

Java文件前后端上传下载工具类

任何非压缩格式下载 package com.pisx.pd.eco.util;import java.io.*; import java.util.Collections; import java.util.HashMap; import java.util.Map;import javax.servlet.ServletOutputStream; import javax.servlet.http.HttpServletResponse;import org.springframewo…

C++11——包装器与lambda表达式

目录 一.背景 二.lambda 1.见一见lambda 2.lambda表达式语法 3.lambda捕捉列表说明 三.函数对象与lambda表达式 四.包装器 1.function包装器 2.包装类的成员函数 五.bind 1.调整参数位置 2.减少函数参数 一.背景 在C98中&#xff0c;如果想要对一个数据集合中的元素…

成都瀚网科技:如何有效运营抖店来客呢?

随着电子商务的快速发展和移动互联网的普及&#xff0c;越来越多的企业开始将目光转向线上销售渠道。其中&#xff0c;抖音成为备受关注的平台。作为中国最大的短视频社交平台之一&#xff0c;抖音每天吸引数亿用户&#xff0c;这也为企业提供了巨大的商机。那么&#xff0c;如…

Integer包装类常用方法和属性

包装类 什么是包装类Integer包装类常用方法和属性 什么是包装类 Java 包装类是指为了方便处理基本数据类型而提供的对应的引用类型。Java 提供了八个基本数据类型&#xff08;boolean、byte、short、int、long、float、double、char&#xff09;&#xff0c;每个基本数据类型对…