Spring事件,ApplicationEvent在业务中的应用
前言
关于事件驱动模型,百度百科在有明确的解释。在
JDK
的Util
包里抽象了事件驱动,有兴趣的朋友可以自行去看下相关类的定义。Spring事件模型ApplicationEvent
是基于JDK
里的事件模型,废话不多说,直接看Spring是如何定义事件模型,以及在具体业务场景中的应用。
事件
事件就是事件,鼠标点击一下算一个事件,某个按钮被点击了一下算一个点击事件,那么我订单支付了可以认为支付也算一个件事!触发了某个事件...... 等等。
抽象类ApplicationEvent
承载着我们要传播的事件或者消息,白话就是说可以把某个对象用ApplicationEvent
这个对象来里的Source
来引用。
监听者
上面我们定义了事件,那么事件产生的一系列的效应或者是变动,那么都由这些监听者们去实现。点击下鼠标(事件),那么我记录下日志,你弹出个提示框。支付某个订单(事件),我记录下记录,他发送个支付通知...... 等等。
泛型接口ApplicationListener
规定了泛型E的上边界为ApplicationEvent
,意思很明确,就是给我们自定义事件用的。Spring
最大的优点我认为是留给用户发挥的空间很大,就像神秘的海洋一样,它一直有你探索不完的秘密,每一次你去了解它,它都能给你带来新的事物和理解。
实战
文章SpringPlugin-Core在业务中的应用中,我们用SpringPlugin
插件的方式去实现了订单的不同操作!而在某个操作里面,我们可能又要发送操作事件的通知,比如:订单支付了后,要通知打印机打印小票、微信公众号提醒支付信息等等。那么我们来实际的操作下。
定义事件源
public class OrderPayedEvent extends ApplicationEvent {
/**
* 消息体,这里就设定为当前订单对象
*/
private final Order order;
public OrderPayedEvent(Object source) {
super(source);
this.order = (Order) source;
}
public Order getOrder() {
return order;
}
}
实现ApplicationEvent
, 我这里Source
实际传递就是Order
对象,当然你也可以定义其他的多参数构造函数!
定义监听者
定义监听者的方式,Spring
提供了两种,一种是接口方式,一种是注解方式。
接口方式
@Component
@Order(1)
public class OrderPayedPrinterListener implements ApplicationListener<OrderPayedEvent> {
@Override
public void onApplicationEvent(OrderPayedEvent event) {
System.out.printf("【线程 - %s 】订单成功成功:第一步,打印小票%n", Thread.currentThread().getName());
}
}
@Component
@Order(2)
public class OrderPayedSendMessageListener implements ApplicationListener<OrderPayedEvent> {
@Override
public void onApplicationEvent(OrderPayedEvent event) {
System.out.printf("【线程 - %s 】订单成功成功:第二步,发送通知商品中心添加库存%n", Thread.currentThread().getName());
}
}
这里我定义了两个监听者,实现泛型接口ApplicationListener
类型为我们刚定义的OrderPayedEvent
这里加上Order
注解,是因为我们有多个监听者,有此业务场景中可能会有顺序的要求!
注解方式
@Component
public class OrderPayListener {
@EventListener(classes = {OrderPayedEvent.class})
public void sendTips(OrderPayedEvent event) {
System.out.printf("【线程 - %s 】订单成功成功:发送用户订单支付消息%n", Thread.currentThread().getName());
}
@EventListener(classes = {OrderPayedEvent.class})
public void reward(OrderPayedEvent event) {
System.out.printf("【线程 - %s 】订单成功成功:奖励业务%n", Thread.currentThread().getName());
}
}
两种方式,各有千秋,不同业务场景选择不同实现方式即可。但注解方式是不会有排序功能的,如果你有业务有需要排序,那么建议换成接口方式
发布件事
万事具备,只欠东风。那么只要合适的位置发布事件即可,那么在上回文章中的支付成功代码加上事件即可
@Component
public class PayOperator implements OrderOperatorPlugin {
//这里注入 应用上下文,可以注入 applicationEventPublisher
@Resource
ApplicationContext context;
// @Resource
// ApplicationEventPublisher applicationEventPublisher;
@Override
public Optional<?> apply(OrderOperatorDTO operator) {
//支付操作
//doPay()
//发送事件
context.publishEvent(new OrderPayedEvent(new Order()));
return Optional.of("支付成功");
}
@Override
public boolean supports(OrderOperatorDTO operatorDTO) {
return operatorDTO.getOperatorType() == OrderOperatorType.PAY;
}
}
打印如下:
【线程 - main 】订单成功成功:第一步,打印小票
【线程 - main 】订单成功成功:第二步,发送通知商品中心添加库存
【线程 - main 】订单成功成功:第四步,奖励业务
【线程 - main 】订单成功成功:第三步,发送用户订单支付消息
那么,ApplicationEvent
对异步支持是怎么样的呢?
只要在启动类上加上@EnableAsync
,在方法体加上@Async
再打印如下:
【线程 - task-1 】订单成功成功:第一步,打印小票
【线程 - task-2 】订单成功成功:第二步,发送通知商品中心添加库存
【线程 - task-4 】订单成功成功:第四步,奖励业务
【线程 - task-3 】订单成功成功:第三步,发送用户订单支付消息
总结
不管是EventObject
,还是Observable
模型,都是用来解耦代码。高内聚,低耦合的设计思想一至到现在都没有被突破过,也是我们在日常工作过程中时刻要提醒自己的编码思想。而我们更要利用好这些前人留下的精髓,应用到我们实际的业务场景中去。
[代码在GitHub](
Spring事件,ApplicationEvent在业务中的应用的更多相关文章
- 从spring源码汲取营养:模仿spring事件发布机制,解耦业务代码
前言 最近在项目中做了一项优化,对业务代码进行解耦.我们部门做的是警用系统,通俗的说,可理解为110报警.一条警情,会先后经过接警员.处警调度员.一线警员,警情是需要记录每一步的日志,是要可追溯的,比 ...
- redis整合Spring集群搭建及业务中的使用
1.redis安装 Redis是c语言开发的. 安装redis需要c语言的编译环境.如果没有gcc需要在线安装.yum install gcc-c++ 安装步骤: 第一步:redis的源码包上传到li ...
- spring事件机制
前置知识补充: 程序里面所谓的“上下文”就是程序的执行环境,打个比方:你就相当于web程序,你的房子就相当于web程序的上下文,你可以在家里放东西,也可以取东西,你的衣食住行都依赖这个房子,这个房子就 ...
- Spring事件发布与监听机制
我是陈皮,一个在互联网 Coding 的 ITer,微信搜索「陈皮的JavaLib」第一时间阅读最新文章,回复[资料],即可获得我精心整理的技术资料,电子书籍,一线大厂面试资料和优秀简历模板. 目录 ...
- 淘系工程师讲解的使用Spring特性优雅书写业务代码
使用Spring特性优雅书写业务代码 大家在日常业务开发工作中相信多多少少遇到过下面这样的几个场景: 当某一个特定事件或动作发生以后,需要执行很多联动动作,如果串行去执行的话太耗时,如果引入消息中 ...
- spring发布和接收定制的事件(spring事件传播)
spring发布和接收定制的事件(spring事件传播) 2012-12-26 20:05 22111人阅读 评论(2) 收藏 举报 分类: 开源技术(如Struts/spring/Hibernat ...
- 【Spring】关于Boot应用中集成Spring Security你必须了解的那些事
Spring Security Spring Security是Spring社区的一个顶级项目,也是Spring Boot官方推荐使用的Security框架.除了常规的Authentication和A ...
- Spring事件解析
首先介绍Spring事件相关类的关系: 其中EventListener与EventObject均是Java SE的范畴,源码如下: package java.util; public interfac ...
- 【转】Spring总结以及在面试中的一些问题
[转]Spring总结以及在面试中的一些问题. 1.谈谈你对spring IOC和DI的理解,它们有什么区别? IoC Inverse of Control 反转控制的概念,就是将原本在程序中手动创建 ...
随机推荐
- 恶意软件开发——编写第一个Loader加载器
一.什么是shellcode loader? 上一篇文章说了,我们说到了什么是shellcode,为了使我们的shellcode加载到内存并执行,我们需要shellcode加载器,也就是我们的shel ...
- MySQL-Cluster 初识
最近,对mysql-cluster进行初步了解,发现和oracle提供的RAC有一定的相似之处,但区别又很大,下面主要是mysql-cluster的搭建,至于对其的深入了解,留着以后工作需 ...
- springMVC学习总结(二) --springMVC表单处理、标签库、静态文件处理
根据springMVC学习总结(一) --springMVC搭建 搭建项目 一.表单处理 1.创建两个java类 Student.java, StudentController.java. 2.在js ...
- 剑指 Offer 38. 字符串的排列
剑指 Offer 38. 字符串的排列 输入一个字符串,打印出该字符串中字符的所有排列. 你可以以任意顺序返回这个字符串数组,但里面不能有重复元素. 示例: 输入:s = "abc" ...
- [编译] 10、kconfig 入门指导教程
目录 前言 1. 安装 kconfig 2. 克隆一个 demo 3. 运行 kconfig 4. 源码解析 4.1 选择题目设计模板 4.2 填空题目设计模板 4.3 判断题目设计模板 5. 产物解 ...
- Linux - centos7.X 安裝 Python 3.7
说明 全部操作都在 root 用户下执行 安装编译相关工具 yum -y groupinstall "Development tools" yum -y install zlib- ...
- Hive的分桶表
[分桶概述] Hive表分区的实质是分目录(将超大表的数据按指定标准细分到指定目录),且分区的字段不属于Hive表中存在的字段:分桶的实质是分文件(将超大文件的数据按指定标准细分到分桶文件),且分桶的 ...
- 《openssl编程》:第一章基础知识
第一章 基础知识 1.1 对称算法 对称算法使用一个密钥.给定一个明文和一个密钥,加密产生密文,其长度和明文大致相同.解密时,使用读密钥与加密密钥相同. 对称算法主要有四种加密模式: (1) 电子密码 ...
- UVA 506 System Dependencies(模拟 烂题)
https://vjudge.net/problem/UVA-506 题目是给出了五种指令,DEPEND.INSTALL.REMOVE.LIST.END,操作的格式及功能如下: DEPEND item ...
- 洛谷P1603——斯诺登的密码(字符串处理)
https://www.luogu.org/problem/show?pid=1603#sub 题目描述 2013年X月X日,俄罗斯办理了斯诺登的护照,于是他混迹于一架开往委内瑞拉的飞机.但是,这件事 ...