波哥带你探寻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

相关文章

Java编译生成多个.class文件的原理和作用

《Java编译生成多个.class文件的原理和作用》作为一名经验丰富的开发者,在Java项目中执行编译后,可能会发现一个.java源文件有时会产生多个.class文件,从技术实现层面详细剖析这一现象... 目录一、内部类机制与.class文件生成成员内部类(常规内部类)局部内部类(方法内部类)匿名内部类二、

SpringBoot实现数据库读写分离的3种方法小结

《SpringBoot实现数据库读写分离的3种方法小结》为了提高系统的读写性能和可用性,读写分离是一种经典的数据库架构模式,在SpringBoot应用中,有多种方式可以实现数据库读写分离,本文将介绍三... 目录一、数据库读写分离概述二、方案一:基于AbstractRoutingDataSource实现动态

Springboot @Autowired和@Resource的区别解析

《Springboot@Autowired和@Resource的区别解析》@Resource是JDK提供的注解,只是Spring在实现上提供了这个注解的功能支持,本文给大家介绍Springboot@... 目录【一】定义【1】@Autowired【2】@Resource【二】区别【1】包含的属性不同【2】@

springboot循环依赖问题案例代码及解决办法

《springboot循环依赖问题案例代码及解决办法》在SpringBoot中,如果两个或多个Bean之间存在循环依赖(即BeanA依赖BeanB,而BeanB又依赖BeanA),会导致Spring的... 目录1. 什么是循环依赖?2. 循环依赖的场景案例3. 解决循环依赖的常见方法方法 1:使用 @La

Java枚举类实现Key-Value映射的多种实现方式

《Java枚举类实现Key-Value映射的多种实现方式》在Java开发中,枚举(Enum)是一种特殊的类,本文将详细介绍Java枚举类实现key-value映射的多种方式,有需要的小伙伴可以根据需要... 目录前言一、基础实现方式1.1 为枚举添加属性和构造方法二、http://www.cppcns.co

Elasticsearch 在 Java 中的使用教程

《Elasticsearch在Java中的使用教程》Elasticsearch是一个分布式搜索和分析引擎,基于ApacheLucene构建,能够实现实时数据的存储、搜索、和分析,它广泛应用于全文... 目录1. Elasticsearch 简介2. 环境准备2.1 安装 Elasticsearch2.2 J

Java中的String.valueOf()和toString()方法区别小结

《Java中的String.valueOf()和toString()方法区别小结》字符串操作是开发者日常编程任务中不可或缺的一部分,转换为字符串是一种常见需求,其中最常见的就是String.value... 目录String.valueOf()方法方法定义方法实现使用示例使用场景toString()方法方法

Java中List的contains()方法的使用小结

《Java中List的contains()方法的使用小结》List的contains()方法用于检查列表中是否包含指定的元素,借助equals()方法进行判断,下面就来介绍Java中List的c... 目录详细展开1. 方法签名2. 工作原理3. 使用示例4. 注意事项总结结论:List 的 contain

Java实现文件图片的预览和下载功能

《Java实现文件图片的预览和下载功能》这篇文章主要为大家详细介绍了如何使用Java实现文件图片的预览和下载功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... Java实现文件(图片)的预览和下载 @ApiOperation("访问文件") @GetMapping("

Spring Boot + MyBatis Plus 高效开发实战从入门到进阶优化(推荐)

《SpringBoot+MyBatisPlus高效开发实战从入门到进阶优化(推荐)》本文将详细介绍SpringBoot+MyBatisPlus的完整开发流程,并深入剖析分页查询、批量操作、动... 目录Spring Boot + MyBATis Plus 高效开发实战:从入门到进阶优化1. MyBatis