波哥带你探寻SpringBoot中优雅设计监听器的本质

2023-11-08 23:50

本文主要是介绍波哥带你探寻SpringBoot中优雅设计监听器的本质,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

请添加图片描述

SpringBoot源码之监听器设计

1.观察者模式

  监听器的设计会使用到Java设计模式中的观察者模式,所以在搞清楚SpringBoot中的监听器的设计之前我们还是非常有必要把观察者模式先弄清楚。

  观察者模式又称为发布/订阅(Publish/Subscribe)模式,在对象之间定义了一对多的依赖,这样一来,当一个对象改变状态,依赖它的对象会收到通知并自动更新.

  在java.util包中包含有基本的Observer接口和Observable抽象类.功能上和Subject接口和Observer接口类似.不过在使用上,就方便多了,因为许多功能比如说注册,删除,通知观察者的那些功能已经内置好了.

1.1 定义具体被观察者

package com.dpb.observer2;import java.util.Observable;/*** 目标对象* 继承 Observable* @author dengp**/
public class ConcreteSubject extends Observable {private int state; public void set(int s){state = s;  //目标对象的状态发生了改变setChanged();  //表示目标对象已经做了更改notifyObservers(state);  //通知所有的观察者}public int getState() {return state;}public void setState(int state) {this.state = state;}
}

观察者只需要继承Observable父类。发送消息的方式执行如下两行代码即可

setChanged();  //表示目标对象已经做了更改
notifyObservers(state);  //通知所有的观察者

Observable源码对应的是:
在这里插入图片描述
在这里插入图片描述

1.2 定义具体观察者

package com.dpb.observer2;import java.util.Observable;
import java.util.Observer;
/*** 观察者模式:观察者(消息订阅者)* 实现Observer接口* @author dengp**/
public class ObserverA implements Observer {private int myState;@Overridepublic void update(Observable o, Object arg) {myState = ((ConcreteSubject)o).getState();}public int getMyState() {return myState;}public void setMyState(int myState) {this.myState = myState;}
}

观察者也就是订阅者只需要实现Observer接口并重写相关update方法即可,在目标实现中我们发现触发的时候执行的就是观察者的update方法。

1.3 测试

package com.dpb.observer2;public class Client {public static void main(String[] args) {//创建目标对象ObserableConcreteSubject subject = new ConcreteSubject();//创建观察者ObserverA obs1 = new ObserverA();ObserverA obs2 = new ObserverA();ObserverA obs3 = new ObserverA();//将上面三个观察者对象添加到目标对象subject的观察者容器中subject.addObserver(obs1);subject.addObserver(obs2);subject.addObserver(obs3);//改变subject对象的状态subject.set(3000);System.out.println("===============状态修改了!");//观察者的状态发生了变化System.out.println(obs1.getMyState());System.out.println(obs2.getMyState());System.out.println(obs3.getMyState());subject.set(600);System.out.println("===============状态修改了!");//观察者的状态发生了变化System.out.println(obs1.getMyState());System.out.println(obs2.getMyState());System.out.println(obs3.getMyState());//移除一个订阅者subject.deleteObserver(obs2);subject.set(100);System.out.println("===============状态修改了!");//观察者的状态发生了变化System.out.println(obs1.getMyState());System.out.println(obs2.getMyState());System.out.println(obs3.getMyState());}
}

在这里插入图片描述
  这样就实现了官方提供观察者模式.

2.SpringBoot中监听器的设计

  然后我们来看下SpringBoot启动这涉及到的监听器这块是如何实现的。

2.1 初始化操作

  通过前面的介绍我们知道在SpringApplication的构造方法中会加载所有声明在spring.factories中的监听器。

image.png

  通过Debug模式我们可以看到加载的监听器有哪些。

image.png

  其实就是加载的spring.factories文件中的key为ApplicationListener的value

image.png

image.png

  通过对这些内置监听器的源码查看我们发现这些监听器都实现了 ApplicationEvent接口。也就是都会监听 ApplicationEvent发布的相关的事件。ApplicationContext事件机制是观察者设计模式的实现,通过ApplicationEvent类和ApplicationListener接口,可以实现ApplicationContext事件处理。

image.png

2.2 run方法

  然后我们来看下在SpringApplication.run()方法中是如何发布对应的事件的。

image.png

  首先会通过getRunListeners方法来获取我们在spring.factories中定义的SpringApplicationRunListener类型的实例。也就是EventPublishingRunListener.

image.png

image.png

image.png

  加载这个类型的时候会同步的完成实例化。

image.png

image.png

  实例化操作就会执行EventPublishingRunListener.

image.png

  在这个构造方法中会绑定我们前面加载的11个过滤器。

image.png

  到这其实我们就已经清楚了EventPublishingRunListener和我们前面加载的11个监听器的关系了。然后在看事件发布的方法。

image.png

查看starting()方法。

image.png

再进入

image.png

进入到multicastEvent中方法中我们可以看到具体的触发逻辑

image.png

在这儿以ConfigFileApplicationListener为例。

image.png

触发会进入ConfigFileApplicationListener对象的onApplicationEvent方法中,

image.png

通过代码我们可以发现当前的事件是ApplicationStartingEvent事件,都不满足,所以ConfigFileApplicationListener在SpringBoot项目开始启动的时候就不会做任何的操作。而当我们在配置环境信息的时候,会发布对应的事件来触发

image.png

image.png

继续进入

image.png

继续进入

image.png

然后再触发ConfigFileApplicationListener监听器的时候就会触发如下方法了

image.png

  其实到这儿,后面的事件发布与监听器的处理逻辑就差不多是一致了。到这儿对应SpringBoot中的监听器这块就分析的差不错了。像SpringBoot的属性文件中的信息什么时候加载的就是在这些内置的监听器中完成的。

image.png

官方内置的事件有:

image.png

  好了本文就给大家介绍到这里,希望能对你有所帮助哦。
请添加图片描述

这篇关于波哥带你探寻SpringBoot中优雅设计监听器的本质的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/373148

相关文章

Spring boot整合dubbo+zookeeper的详细过程

《Springboot整合dubbo+zookeeper的详细过程》本文讲解SpringBoot整合Dubbo与Zookeeper实现API、Provider、Consumer模式,包含依赖配置、... 目录Spring boot整合dubbo+zookeeper1.创建父工程2.父工程引入依赖3.创建ap

SpringBoot结合Docker进行容器化处理指南

《SpringBoot结合Docker进行容器化处理指南》在当今快速发展的软件工程领域,SpringBoot和Docker已经成为现代Java开发者的必备工具,本文将深入讲解如何将一个SpringBo... 目录前言一、为什么选择 Spring Bootjavascript + docker1. 快速部署与

Spring Boot spring-boot-maven-plugin 参数配置详解(最新推荐)

《SpringBootspring-boot-maven-plugin参数配置详解(最新推荐)》文章介绍了SpringBootMaven插件的5个核心目标(repackage、run、start... 目录一 spring-boot-maven-plugin 插件的5个Goals二 应用场景1 重新打包应用

SpringBoot+EasyExcel实现自定义复杂样式导入导出

《SpringBoot+EasyExcel实现自定义复杂样式导入导出》这篇文章主要为大家详细介绍了SpringBoot如何结果EasyExcel实现自定义复杂样式导入导出功能,文中的示例代码讲解详细,... 目录安装处理自定义导出复杂场景1、列不固定,动态列2、动态下拉3、自定义锁定行/列,添加密码4、合并

Spring Boot集成Druid实现数据源管理与监控的详细步骤

《SpringBoot集成Druid实现数据源管理与监控的详细步骤》本文介绍如何在SpringBoot项目中集成Druid数据库连接池,包括环境搭建、Maven依赖配置、SpringBoot配置文件... 目录1. 引言1.1 环境准备1.2 Druid介绍2. 配置Druid连接池3. 查看Druid监控

Java中读取YAML文件配置信息常见问题及解决方法

《Java中读取YAML文件配置信息常见问题及解决方法》:本文主要介绍Java中读取YAML文件配置信息常见问题及解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要... 目录1 使用Spring Boot的@ConfigurationProperties2. 使用@Valu

创建Java keystore文件的完整指南及详细步骤

《创建Javakeystore文件的完整指南及详细步骤》本文详解Java中keystore的创建与配置,涵盖私钥管理、自签名与CA证书生成、SSL/TLS应用,强调安全存储及验证机制,确保通信加密和... 目录1. 秘密键(私钥)的理解与管理私钥的定义与重要性私钥的管理策略私钥的生成与存储2. 证书的创建与

浅析Spring如何控制Bean的加载顺序

《浅析Spring如何控制Bean的加载顺序》在大多数情况下,我们不需要手动控制Bean的加载顺序,因为Spring的IoC容器足够智能,但在某些特殊场景下,这种隐式的依赖关系可能不存在,下面我们就来... 目录核心原则:依赖驱动加载手动控制 Bean 加载顺序的方法方法 1:使用@DependsOn(最直

SpringBoot中如何使用Assert进行断言校验

《SpringBoot中如何使用Assert进行断言校验》Java提供了内置的assert机制,而Spring框架也提供了更强大的Assert工具类来帮助开发者进行参数校验和状态检查,下... 目录前言一、Java 原生assert简介1.1 使用方式1.2 示例代码1.3 优缺点分析二、Spring Fr

java使用protobuf-maven-plugin的插件编译proto文件详解

《java使用protobuf-maven-plugin的插件编译proto文件详解》:本文主要介绍java使用protobuf-maven-plugin的插件编译proto文件,具有很好的参考价... 目录protobuf文件作为数据传输和存储的协议主要介绍在Java使用maven编译proto文件的插件