解耦神器Event和EventListener

总结版

说到解耦,就不得不说经典的订阅和发布者模式,我们当然可以在代码中自己实现一套这样的模式,不过spring中已经帮忙我们实现好了现成的,直接拿来用即可。

即发布事件时用

scss 复制代码
ApplicationContextHelper.getApplicationContext().publishEvent(Event event)

监听时用

typescript 复制代码
@EventListener
public void onEvent(Event event){
//自己的业务逻辑,相关参数可以放在event中
}

另外,在spring4.2+中,还额外提供 @TransactionalEventListener注解,进行升级,可以在event的各个事务阶段进行监听。

问题

许多的系统都会涉及到订单,在基金系统中也不例外。一开始订单提交完成之后,动作很简单,打印订单就可以了,不过随着功能越来越多,后面的动作也越来越多,加入了发送通知邮件,冻结,解冻,生成订单文件等等,

这个类里面的placeOrder的方法也越来越长,不加约束时轻易超过上千行,可读性极差,也极难再做扩展。

csharp 复制代码
public void placeOrder(){
   //...上千行内容,所有逻辑全写在一起
}

后面抽象出来一部分动作,但是内容也时常变动

scss 复制代码
public void placeOrder(){
    //...正常订单逻辑
    emailService.sendEmail();
    financialService.freeze();
    financialService.unfreeze();
    fileService.generateOrderFile();
}

那么有没有什么其他的方法呢?

方案

后面我们用到发布和监听模式:

csharp 复制代码
public void placeOrder(){
   //...正常订单逻辑
   applicationContext.publishEvent(Event orderPlacedEvent);
}

这个主体定下来之后便很少再做变动,使整个方法的职责尽可能地少,只关注订单本身的逻辑。之后我们再依据需求,对这个event进行监听。

less 复制代码
@Component
@EnableAsync
public class EmailListener{

  @EventListener
  @Async
  @Order(1)
  public void onEvent(Event event){
     //不同的监听器可以放在不同的类中,使类的职责尽可能地小比
     //比如从此处理邮件发送,他处处理冻结解冻
     emailService.sendEmail();
  }
}

红色部分可以额外的定义是否同步监听,同步监听可以认为是放在了一个方法内(如果有数据库事务,会默认加入同一事务),@Order注解会决定监听的顺序。

另外也可以使用 @TransactionalEventListener 替换 @EventListener, 比如,我需要在之前的order被正确的存入数据库(事务提交成功时),才执行监听:

less 复制代码
@TransactionalEventListener(phase = TransactionPhase.AFTER_COMMIT, 
classes = Event .class)
@Async
public void onEvent(Event event){
  //...处理内容
}

这个时候注意如果阶段是phase = TransactionPhase.AFTER_COMMIT,那么必须得是异步,否则,语义是"等我自身被提交,我再去做监听",会是一个矛盾的内容。

思考

我们的思路,始终是围绕一个主题,即如何让代码能够经过最小的改动,牵涉最少的其他模块,去进行新需求的编写。

这也是为什么一些设计模式那么经典的原因。

相关推荐
一定要AK6 小时前
Spring 入门核心笔记
java·笔记·spring
A__tao6 小时前
Elasticsearch Mapping 一键生成 Java 实体类(支持嵌套 + 自动过滤注释)
java·python·elasticsearch
KevinCyao6 小时前
java视频短信接口怎么调用?SpringBoot集成视频短信及回调处理Demo
java·spring boot·音视频
迷藏4946 小时前
**发散创新:基于Rust实现的开源合规权限管理框架设计与实践**在现代软件架构中,**权限控制(RBAC)** 已成为保障
java·开发语言·python·rust·开源
wuxinyan1237 小时前
Java面试题47:一文深入了解Nginx
java·nginx·面试题
新知图书7 小时前
搭建Spring Boot开发环境
java·spring boot·后端
冰河团队7 小时前
一个拉胯的分库分表方案有多绝望?整个部门都在救火!
java·高并发·分布式数据库·分库分表·高性能
洛_尘7 小时前
Java EE进阶:Linux的基本使用
java·java-ee
宸津-代码粉碎机7 小时前
Spring Boot 4.0虚拟线程实战调优技巧,最大化发挥并发优势
java·人工智能·spring boot·后端·python
MaCa .BaKa7 小时前
47-心里健康咨询平台/心理咨询系统
java·spring boot·mysql·tomcat·maven·intellij-idea·个人开发