Spring源码之事件监听机制(下)

2023-10-28


前言

这篇文章接的是上篇文章Spring源码之事件监听机制(上)来讲,通过理解Spring内部的监听机制,手写一个类似的监听机制框架,再从中抽象出设计模式。


一、手写事件监听机制框架

1.准备

JDK为我们提供了事件监听的支持,在java.util包的下面提供了事件监听接口以及事件类。
JDK监听接口
JDK事件类

2.事件监听接口

public interface IEventListener<E extends EventObject> extends EventListener {

    void onApplicationEvent(E event);
}

定义一个事件监听接口供事件监听类实现

3.事件管理器

public class EventMulticaster {

    private Set<IEventListener> set = new LinkedHashSet<>();

    public void registerListener(IEventListener listener) {
        if (!set.contains(listener)) {
            set.add(listener);
        }
    }

    public void removeListener(IEventListener listener) {
        if (!set.contains(listener)) {
            set.remove(listener);
        }
    }

    public Set<IEventListener> getEventListeners() {
        return set;
    }

}

定义一个事件管理器,存储事件监听以及对事件监听的增删操作

4.事件发布器

public class EventPublisher {

    private EventMulticaster eventMulticaster;

    public void initEventMulticaster() {
        if (eventMulticaster == null) {
            eventMulticaster = new EventMulticaster();
        }
    }

    public void registerListener(IEventListener listener) {
        if (eventMulticaster != null) {
            eventMulticaster.registerListener(listener);
        }
    }

    public void removeListener(IEventListener listener) {
        if (eventMulticaster != null) {
            eventMulticaster.removeListener(listener);
        }
    }

    public void publishEvent(EventObject eventObject) {
        if (eventMulticaster != null) {
            Set<IEventListener> set = eventMulticaster.getEventListeners();
            for (IEventListener listener : set) {
                listener.onApplicationEvent(eventObject);
            }
        }
    }
}

5.需求

和上一篇一样:在订单服务中,用户下单成功后,需要物流服务和库存服务进行相应的处理,采取异步解耦的方式。

6.编码

  1. 订单事件
public class OrderEvent extends EventObject {

    public OrderEvent(Object source) {
        super(source);
    }
}
  1. 库存监听
public class StoreListener implements IEventListener<OrderEvent>{
    @Override
    public void onApplicationEvent(OrderEvent event) {
        System.out.println("库存服务运行中……");
    }
}
  1. 物流监听
public class LogisticsListener implements IEventListener<OrderEvent> {
    @Override
    public void onApplicationEvent(OrderEvent event) {
        System.out.println("物流服务运行中……");
    }
}
  1. 订单服务
System.out.println("订单服务开始运行");
System.out.println("创建订单完成,通知物流、库存……");
// 初始化事件管理器
EventPublisher eventPublisher = new EventPublisher();
eventPublisher.initEventMulticaster();
// 添加事件监听
eventPublisher.registerListener(new LogisticsListener());
eventPublisher.registerListener(new StoreListener());
// 发布事件
eventPublisher.publishEvent(new OrderEvent(this));
  1. 结果

订单服务开始运行
创建订单完成,通知物流、库存……
物流服务运行中……
库存服务运行中……

二、观察者模式

1.概述

观察者模式(有时又被称为模型(Model)-视图(View)模式、源-收听者(Listener)模式或从属者模式)是软件设计模式的一种。在此种模式中,一个目标物件管理所有相依于它的观察者物件,并且在它本身的状态改变时主动发出通知。这通常透过呼叫各观察者所提供的方法来实现。此种模式通常被用来实现事件处理系统。(百度百科)

事件监听机制进行建模分析,当不考虑事件发布器之后,可以简单的理解为:一个事件对应着多个监听者,当事件发布的时候,监听者收到消息,是不是就是典型的观察者模式。

2.UML图

其实JDK已经为我们提供了对观察者模式的支持,下面是我根据JDK中提供的观察者与被观察者绘制的UML图:
观察者模式UML图
从UML图中可以看出:被观察者也被称为主题,与观察者之间是一对多的关系,主题存储着所有观察者,当主题状态变更的时候,通知观察者进行更新。

3.Coding验证

  1. 主题
public class Subject extends Observable {

    public void setChanged() {
        super.setChanged();
    }
}
  1. 观察者
public class ConcreteObserver1 implements Observer {

    private String name;

    public ConcreteObserver1(String name) {
        this.name = name;
    }

    @Override
    public void update(Observable o, Object arg) {
        System.out.println(name + ":当前状态:" + o.hasChanged() + ",内容:" + arg.toString());
    }
}
  1. 客户端验证
public class Client {
    public static void main(String[] args) {
        Subject subject = new Subject();
        subject.addObserver(new ConcreteObserver1("观察者1"));
        subject.addObserver(new ConcreteObserver2("观察者2"));
        subject.setChanged();
        subject.notifyObservers("我已经不是当初你认识的我自己……");
    }
}
  1. 结果

观察者2:当前状态:false,内容:我已经不是当初你认识的我自己……
观察者1:当前状态:false,内容:我已经不是当初你认识的我自己……


小结

  从Spring的事件监听机制,到自定义监听框架,再到观察者模式,一路走来,从繁入简,相信大家对Spring的监听机制已经有了清晰的认知。
  那我们从设计模式,到Sping的事件监听再回顾一下。当对象之间存在着一对多的关系,一个对象改变,依赖于它的所有对象都会收到通知,这就是我们的观察者模式。但被观察者与观察者之间依然存在着很强的耦合关系,所以为了降低这种耦合关系,我们在被观察者和观察者之间放置一个消息通道,也就是事件发布器,这就是事件监听机制,Spring的事件监听机制就是依托于此,只不过它更加的完善。
  好了,这篇文章到这里就结束了,相信到这里再回头看Spring的事件监听机制源码一定会有不一样的收获。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Spring源码之事件监听机制(下) 的相关文章

  • Spring Security PermitAll 不允许匿名访问

    我有一个方法 我想允许匿名和经过身份验证的访问 我正在使用 Spring Security 3 2 4 和基于 Java 的配置 重写的配置方法 在我的自定义配置类中扩展WebSecurityConfigurerAdapter 有以下内容h
  • HTTP请求压缩

    一般用例 想象一下一个正在上传大量 JSON 的客户端 内容类型应保留application json因为这描述了实际数据 Accept Encoding 和 Transfer Encoding 似乎是为了告诉服务器应该如何格式化响应 看起
  • Spring 3.0 SimpleMailMessage 支持

    我在我的 gmail 帐户上收到电子邮件 但发件人始终是我自己 而不是该人的电子邮件地址 即使我在实现类中将其硬编码为message setFrom email protected cdn cgi l email protection 仍然
  • 如何将 Spring Boot 应用程序部署到 Tomcat 6 Servlet 2.5

    我使用 Spring Boot 创建了一个常见问题解答 它需要部署到 tomcat 6 服务器 servlet 2 5 我需要配置当前的父 java 应用程序 war web xml 以将所有请求指向 url 模式 faq 例如 指向我的
  • 执行 POST 请求时 Spring Boot 端点 403 OPTIONS

    我正在使用 Spring 运行一个服务 当我的 Angular 前端尝试发出 POST 请求时 会收到带有请求方法 选项的 403 错误 Spring 服务和 Angular 应用程序都在我的机器上本地运行 我尝试使用 Chrome 插件切
  • javax.servlet.jsp.JspTagException:在语言环境“en_US”的代码“errors.contact.sysadmin.msg”下找不到消息。]

    我已经像这样配置了我的消息资源 tradelc servlet xml
  • Tomcat 热部署到托管多个应用程序的实例

    我的问题是关于部署到托管多个应用程序并托管 Struts Spring 和 Hibernate 的应用程序上下文的 Tomcat 服务器实例 我想在不重新启动 Tomcat 服务器的情况下将更改部署到一个应用程序 举个例子 在我们公司很多时
  • Spring / AOP:在数据库中实现活动日志的最佳方式

    我已经阅读了一些 Spring AOP 教程 并且对相关概念有了一定的熟悉 现在满足我的要求 我需要创建一个活动日志实现 它将在数据库中保存登录用户的活动 范围包括申请服务或在以下情况下创建新用户Admin用户等 在调用任何具有注释的方法时
  • 如何使用 spring data jpa 调用数据库函数?

    我想从 Spring Boot 应用程序调用数据库函数 该函数将返回解密的值 是否可以使用spring data jpa调用数据库函数 如果可以的话怎么办 如果无法使用 spring data jpa 是否有其他方法可以从 Spring B
  • 如何获取 JSF 2 中所有会话范围的 bean?

    据我所知 JSF 将所有会话范围 bean 保存在某种 Map 中 如果我错了 请纠正我 在我的应用程序中 我有一个名为 userDetailsBean 的会话范围 由 Spring 管理并注入到支持 bean 中 bean 是否有可能通过
  • Spring Security SAXParseException

    我正在 Spring 中开发 HelloWorld 应用程序 它工作正常 当我添加弹簧安全性时 问题就出现了 我正在通过 Spring Security 3 这本书来指导自己 部署时删除它 abr 26 2014 2 41 35 PM or
  • Spring MVC VersionResourceResolver / ContentVersionStrategy 在 JSP 中无法正常工作

    我有一个 Spring MVC 4 3 0 应用程序 并已使用 ResourceHandlerRegistry 注册了一个 VersionResourceResolver 并添加了 ContentVersionStrategy 我启用了 R
  • 使用 Hibernate 在 MySQL 中存储字节数组

    我正在尝试保存带有字节数组字段的实体 我在 MySQL 数据库之上使用 Hibernate 和 JPA 这是字段定义 对于嵌入式 H2 数据库来说效果很好 Entity name blob public class Blob Lob Bas
  • Spring Retry 不适用于第二级方法

    Retryable似乎不适用于第二级方法 如sphRemoteCall以下 我看到代理已创建 但永远不会在失败时重试 一旦我搬家 Retryable到第一级方法 例如getSubscriberAccount 它开始工作了 下面的例子 Ser
  • Spring @Retryable 与有状态 Hibernate 对象

    我正在尝试使用 Springs Retryable 让我的服务方法在失败时重试 Retryable backoff Backoff delay 1000 maxAttempts 3 Transactional rollbackFor Thr
  • Java 不提供双向 SSL 客户端证书?

    我正在尝试使用相互 SSL 从 Java Spring Boot 应用程序连接到 NetScaler 端点 我可以使用以下命令通过 OpenSSL 在命令行上按预期进行连接 openssl s client connect xxxx xxx
  • 卡夫卡监听器中的钩子

    kafka 监听消息之前 之后是否有任何类型的钩子可用 使用案例 必须设置MDC关联id才能进行日志溯源 我在寻找什么 之前 之后回调方法 以便可以在进入时设置 MDC 关联 ID 并最终在退出时清除 MDC 编辑后的场景 我将关联 id
  • Spring Boot Batch - 不包括 JobLauncherCommandLineRunner

    我在 Spring Boot 中配置了一个简单的 Spring Batch 作业 类似于弹簧导轨 http spring io guides gs batch processing 在启动时 它会自动检测并调用 JobLauncherCom
  • 如何在 Spring 中将多个映射合并为一个

    我的上下文文件中定义了几个地图 有没有一种方法可以将这些映射组合成一个包含其所有条目的映射 而无需编写 Java 代码 并且无需使用嵌套映射 我正在寻找 Map m new HashMap 的等效项m putAll carMap m put
  • spring中如何使用jackson代替JdkSerializationRedisSerializer

    我在我的一个 Java 应用程序中使用 Redis 并且正在序列化要存储在 Redis 中的对象列表 但是 我注意到使用 RedisTemplate 会使用 JdkSerializationRedisSerializer 相反 我想使用 J

随机推荐

  • GD32替代STM32使用Cube MX的HAL库开发

    目录 一 STM32F103与GD32F103 差别比较 二 GD32使用CubeMX配置 1 配置单片机型号 2 晶振配置 3 其它配置 三 GD32使用Keil配置 1 更改型号为GD32芯片 2 编译下载 四 例程下载链接 一 STM
  • C++之智能指针auto_ptr

    当你在读这篇文章的时候 应该都有这样一个疑问 那就是为什么要使用智能指针 我们先看这样一个示例 include
  • RC4(原理+代码+调用openssl库+报错分析)

    目录 一 原理 1 流密码的基本思想 2 RC4流密码算法的原理 1 初始化数据表S和T 2 初始置换数据表S 密钥调度算法 3 生成密钥流 伪随机数生成算法 二 代码实现 三 调用openssl库实现RC4 1 代码实现 2 调用open
  • 结合Wireshark捕获分组深入理解DNS协议

    一 概述 1 1 DNS 识别主机有两种方式 主机名 IP地址 前者便于记忆 如www yahoo com 但路由器很难处理 主机名长度不定 后者定长 有层次结构 便于路由器处理 但难以记忆 折中的办法就是建立IP地址与主机名间的映射 这就
  • vue3的文档

    四 Vue 3 1 TypeScript 1 动态类型的问题 前面我们讲过 js 属于动态类型语言 例如 function test obj obj 可能只是个字符串 test hello world obj 也有可能是个函数 test g
  • SLAM综述阅读笔记七:Visual and Visual-Inertial SLAM: State of the Art, Classification,and Experimental 2021

    Visual and Visual Inertial SLAM State of the Art Classification and Experimental Benchmarking 作者 Myriam Servi res Val ri
  • IP核的使用之ROM(Vivado)

    存储类IP核 ROM 文章目录 存储类IP核 ROM 一 引言 二 ROM IP核及相关内容扫盲 1 ROM简介 2 ROM的初始化文件介绍 3 分布式ROM和块ROM简介 4 单端口ROM和双端口ROM简介 三 分布式ROM IP核的创建
  • 高斯混合模型(GMM)先验的推断

    GMM先验的优化方程 假设图像降质模型为 Y A X N Y AX N Y AX N 我们希望恢复
  • 腾讯云SA3服务器AMD处理器CPU网络带宽性能详解

    腾讯云AMD服务器SA3实例CPU采用2 55GHz主频的AMD EPYCTM Milan处理器 睿频3 5GHz 搭载最新一代八通道DDR4 内存计算性能稳定 默认网络优化 最高内网收发能力达1900万pps 最高内网带宽可支持100Gb
  • 二极管常见分类及使用

    1 肖特基二极管 1 1概念 肖特基二极管 SBD 不是利用P型半导体与N型半导体接触形成PN结原理制作的 而是利用金属与半导体接触形成的金属 半导体结 肖特基势垒 原理制作的 因此 SBD也称为金属 半导体 接触 二极管或表面势垒二极管
  • 获取上个月的起止时间

    function 日期初始化 alert getStartDate alert getEndDate 获取开始时间 function getStartDate var date new Date var year date getFullY
  • Transform 基础知识

    Transform 变换 是场景中最常打交道的类 用于控制物体的位移 旋转 缩放等功能 Transform Class inherits from Component IEnumerable Position rotation and sc
  • HJ41 称砝码

    题目 HJ41 称砝码 题解 import java util 注意类名必须为 Main 不要有任何 package xxx 信息 public class Main public static void main String args
  • RBAC详解

    RBAC详解 1 RBAC模型的工作原理 2 RBAC模型的实现 3 总结 RBAC模型是一种基于角色的访问控制模型 它定义了一些规则和机制来控制用户对系统资源的访问 在本文中 我们将详细讨论RBAC模型的工作原理 并使用一个数据库示例来说
  • 剑指Offer - 面试题49:丑数

    题目 我们把只包含因子2 3 5的数称为丑数 Ugly Number 求按照从小到大的顺序的第1500个丑数 例如 6 8都是丑数 但14不是 因为它包含因子7 习惯上我们把1当作第一个丑数 分析 暴力法 从1开始每个数字都判断 若是丑数
  • 代码实现 —— 基于 STM32 的可见光通信系统课程设计

    目前课设已完成 2m距离 传输10000个连续数字 每个数字两字节大小 即总共20000个字节160000bit 用时7s 大约2 3万bit s 即22 4kB s 误码率为0 视频演示链接 另外 自己写了一个基于QT的串口上位机 结合U
  • 前端面试题汇总(vue+html基础)最新最全

    一 HTML基础部分 1 什么是盒子模型 重要 在网页中 一个元素占有空间的大小由几个部分构成 其中包括元素的内容 content 元素的内边距 padding 元素的边框 border 元素的外边距 margin 四个部分 这四个部分占有
  • FDbus

    文章目录 介绍 背景 特点 FDBus 中间件模型 FDBus 寻址和组网 Server地址 Server命名和地址分配 name server使用如下规则分配server地址 多主机组网 host server的工作原理 client 与
  • 时序分解

    时序分解 Matlab实现CEEMD互补集合经验模态分解时间序列信号分解 目录 时序分解 Matlab实现CEEMD互补集合经验模态分解时间序列信号分解 效果一览 基本介绍 程序设计 参考资料 效果一览 基本介绍 Matlab实现CEEMD
  • Spring源码之事件监听机制(下)

    文章目录 前言 一 手写事件监听机制框架 1 准备 2 事件监听接口 3 事件管理器 4 事件发布器 5 需求 6 编码 二 观察者模式 1 概述 2 UML图 3 Coding验证 小结 前言 这篇文章接的是上篇文章Spring源码之事件