当前位置: 首页 > news >正文

利用springEvent,进行服务内部领域事件处理

SpringEvent的最佳实践


文章目录

  • SpringEvent的最佳实践
  • 前言
  • 一、领域事件是什么
  • 二、使用步骤
    • 1.事务提交之后才发送domainEvent
    • 2.异步处理事件
  • 总结


前言

相信java boy对于Spring Event,多多少少都有一些熟悉了。如果不了解的,可以参考我的这篇文章:Spring的Event编程以及实现原理


一、领域事件是什么

简单说领域事件,其实就是对PO的封装。我这边要说明的并不是DDD对系统落地的改造。而是借鉴DDD的思想,对用户操作的实体-entity,进行封装成一个domainEvent结构。有了这个domainEvent,有两个好处:

  1. 可以对entity进行防腐。该domainEvent作为后续操作的dto,不关注如何通过entity构建的domainEvent,只需要基于该domainEvent进行操作即可
  2. 可以提高并行开发的效率。对entity操作的代码和后续操作的代码,可以让不同的研发同时进行
    领域事件解耦开发

二、使用步骤

针对B端,流量并没有这么大的服务,没有必要考虑分布式事务。但是接入一个自产自销的MQ,感觉又有点过重。还不如SpringEvent来自己处理

1.事务提交之后才发送domainEvent

我们需要考虑的问题是,对于entity进行的操作,一定是写库操作,必然涉及到事务。针对业务流程来说,必然需要保证事务提交成功之后,再进行后续操作

此时可以使用Spring事务的钩子函数,让事务提交成功之后才可以进行操作。

@Transactional(rollbackFor = Exception.class)
public void doProcess(User user) {
// 1.根据entity构建domainEvent的dto,作为防腐层 
UserMsgBO userMsgBO = buildMsgBO(user);
// 2. 注册事务钩子函数,让事务提交之后,进行Event的publish
TransactionSynchronizationManager.registerSynchronization(new TransactionSynchronization() {@Overridepublic void afterCommit() {log.info("publish event, userMsgBO:{}", JSON.toJSONString(userMsgBO));applicationEventPublisher.publishEvent(userMsgBO);}});
}
//3. 数据库操作

2.异步处理事件

需要考虑日志打印问题

根据Spring的Event编程以及实现原理文章里写的,如果要进行事件处理的异步进行,需要进行实例化一个SimpleApplicationEventMulticaster,并设置其taskExecutor属性。实际上呢,还可以使用@Async注解来实现:@Async的使用及原理

@Async("springEventExecutor")
@EventListener(UserMsgBO.class)
public void doSendMsg(UserMsgBO userMsgBO) {log.info("doSendMsg, msg:[{}]", JSON.toJSONString(userMsgBO));try {// 具体的处理逻辑} catch (Exception e) {log.error("doSendMsg error, msg:[{}]", JSON.toJSONString(userMsgBO), e);}
}

可以看到,此时给@Async注解传入了一个线程池:springEventExecutor。还差最后一步,怎么第二节提出的问题,使用了线程池实现了异步,那么ThreadLocal里的traceId是不是丢了呢?

相信大家应该都知道阿里的TransmittableThreadLocal,也知道具体的场景

@Bean("springEventExecutor")
public Executor springEventExecutor() {ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(10, 10, 60L, TimeUnit.SECONDS,new LinkedBlockingQueue<>(1000),new ThreadFactoryBuilder().setNameFormat("spring-event-thread-pool-%d").build(),new ThreadPoolExecutor.CallerRunsPolicy());// 利用TtlExecutors包装一下线程池,可以在日志中打印traceIdreturn TtlExecutors.getTtlExecutorService(threadPoolExecutor);
}

当然保险起见,还可以针对消费domainEvent的doSendMsg方法增加一个后门,防止服务重启时打断异步流程。
当然,如果对于数据一致性要求比较高,那么就借鉴B端系统的最终一致性方案了:可以将domainEvent进行持久化,然后通过定时任务来保证最终一致性。


总结

B端系统开发,需要考虑:

  1. 事务
  2. 兼顾性能

没有银子弹,本文只是给出了一种常见场景的常见解决方案。

http://www.lqws.cn/news/595513.html

相关文章:

  • 使用 icinga2 写入 TDengine
  • 用lines_gauss的width属性提取缺陷
  • C++ 11 中 condition_variable 的探索与实践
  • python 调用C/C++动态库
  • Apache HTTP Server部署全攻略
  • 用于快速训练收敛的 Conditional DETR
  • 基于.Net的Web API 控制器及方法相关注解属性
  • 数字ic后端设计从入门到精通10(含fusion compiler, tcl教学)静态时序分析
  • 3D 商品展示与 AR 试戴能为珠宝行业带来一些便利?
  • 什么是测试用例?它的核心要素有哪些?
  • docker desktop部署本地gitlab服务
  • 【仿muduo库实现并发服务器】Connection模块
  • 【仿muduo库实现并发服务器】Acceptor模块
  • 笔记/计算机网络
  • ChatGPT使用限额记录与插件统计
  • Softhub软件下载站实战开发(九):编写软件配置管理界面
  • Electron 应用打包与分发:从开发到交付的完整指南
  • Call、Apply、Bind详解
  • 如何进行Edge版本回退及禁用更新
  • 结构光相机:重塑工业自动化的“智慧之眼”,驱动智能制造新未来
  • 深度剖析:基于AOP、自定义注解与设计模式构建高度可定制的分布式锁解决方案
  • 亚马逊云科技中国峰会:数新智能CTO原攀峰详解一站式AI原生数智平台DataCyber在Amazon EKS的实践
  • 基于SSM万华城市货运服务系统的设计与实现
  • eNSP实验一:IPv4编址及IPv4路由基础
  • 新手向:从零开始Node.js超详细安装、配置与使用指南
  • 业务系统-AI 智能导航设计(系统设计篇 下)
  • 制作一款打飞机游戏74:游戏原型
  • 【仿muduo库实现并发服务器】LoopThreadPool模块
  • 第八十六篇 大数据排序算法:从厨房整理到分布式排序的智慧
  • 复合型浪涌保护器五大核心技术重构电气防护体系