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源码之事件监听机制(下) 的相关文章

随机推荐

  • 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源码之事件