0


【Spring】从Spring源码入手分析广播与监听并完成项目实战

文章目录

近期疫情形势严峻,情形不容乐观,周末也不敢出去浪了,躲在家里“葛优躺”。闲来无事,又翻了遍

Spring

的源码。不翻不知道,一翻吓一跳,之前翻过的源码已经吃进了肚子里,再见亦是陌生人。

个人建议:为了以后能快速的捡起某个知识点,最好的方法还是形成文档,下次有遗漏的时候,直接读文档,按之前的思路捋一遍,“干净又卫生”。

之前的文章中我们已经介绍过如何在项目中快速上手“事件通知机制”,相信大家已经掌握了。但是我们作为高级

javaer

,要知其然,更要知其所以然。今天就带大家从源码的角度来分析一下广播与监听的底层实现原理。

源码导入教程也给你准备好了,不来试试吗?

版本号:spring-framework-5.0.x

源码解析

为了实现广播与监听的功能,

Spring

为我们提供了两个重要的函数式接口:

ApplicationEventPublisher

ApplicationListener

。前者的

publishEvent()

方法为我们提供了发送广播的能力;后者的

onApplicationEvent()

方法为我们提供了监听并处理事件的能力。

接下来我们就来分析一下

spring

是如何运用这两种能力的。

不知道大家对单例对象的初始化调用过程是否熟悉?主要调用方法流程如下:

发送广播

applyBeanPostProcessorsBeforeInitialization

方法会去遍历该工厂创建的所有的

Bean

后置处理器,然后去依次执行后置处理器对应的

postProcessBeforeInitialization

方法。

在该方法的实现类中我们看到了两个熟悉的类名

不知道大家还记得不,这俩类是在

beanFactory

的准备工作过程中添加的两个

bean

的后置处理器,所以这个地方会依次去执行这两个类中的实现方法。

由于蓝框中类的实现方法是默认实现按照原样返回的给定的

bean

,所以此处不用过多分析,我们重点来看下红框中类的方法实现。

该方法中最重要的是

invokeAwareInterfaces

方法,它的作用是检测对应的

bean

是否实现了某个

Aware

接口,如果实现了的话就去进行相关的调用。

if(bean instanceofApplicationEventPublisherAware){((ApplicationEventPublisherAware) bean).setApplicationEventPublisher(this.applicationContext);}

我们发现在

invokeAwareInterfaces

方法中出现了如上代码,这不就是和广播发送相关的吗?所以只要我们写一个类来实现

ApplicationEventPublisherAware

接口,就可以在该

bean

中注入一个

ApplicationEventPublisher

对象,也就获得了发送广播的能力。

监听消息

applyBeanPostProcessorsAfterInitialization

方法会去遍历该工厂创建的所有的

Bean

后置处理器,然后去依次执行后置处理器对应的

postProcessAfterInitialization

方法。

同样的,该方法的实现类中也有

ApplicationContextAwareProcessor

ApplicationListenerDetector

两个类,但是不同的是,前者的类的实现方法是默认实现按照原样返回的给定

bean

,而后者做了相关的处理。

this.applicationContext.addApplicationListener((ApplicationListener<?>) bean);

上述代码是将实现了

ApplicationListener

接口的

bean

添加到监听器列表中,最终是保存在

AbstractApplicationEventMulticaster

的成员变量

defaultRetriever

的集合

applicationListeners

中。

猜想:当发送广播消息时,就直接找到集合中的这些监听器,然后调用每个监听器的

onApplicationEvent

方法完成事件的处理。

案例分析

refresh()

finishRefresh()

方法中,

publishEvent(newContextRefreshedEvent(this));

发送一条事件类型为

ContextRefreshedEvent

的广播消息,用来代表

Spring

容器初始化结束。通过分析发现,该方法中最主要的就是如下代码:

//真正的广播交给 applicationEventMulticaster 来完成getApplicationEventMulticaster().multicastEvent(applicationEvent, eventType);
refresh()

initApplicationEventMulticaster()

applicationEventMulticaster

初始化为

SimpleApplicationEventMulticaster

在实现类

SimpleApplicationEventMulticaster

的方法中,会找到已注册的

ApplicationListener

列表,然后分别调用

invokeListener

方法(将监听和事件作为参数传到方法并执行的过程就是发送广播的过程)。

底层调用的是

listener.onApplicationEvent(event);

方法,也就是各个监听实现类单独处理广播消息的逻辑。

消息与监听绑定

看到这儿,你是不是已经发现了:消息类型和监听器的绑定发生在广播过程中。接下来就让我们去一探究竟

我们看一下

multicastEvent()

方法中的

getApplicationListeners(event, type)

方法。

在该方法中,用到了

ConcurrentHashMap

类型的缓存

retrieverCache

,所以每种类型的事件在广播的时候会触发一次绑定操作。它的key由事件的来源和类型确定,它的value中就包含了由事件来源和类型所确定的所有监听列表。

其中绑定的逻辑就出现在

retrieveApplicationListeners

方法中,大家可以去源码中查看。

实战教学

纸上得来终觉浅,绝知此事要躬行。为了更好地理解广播与监听的流程,我们当然得用实战来加以辅佐!

自定义事件

publicclassMyEventextendsApplicationContextEvent{publicMyEvent(ApplicationContext source){super(source);}}

自定义广播

@ComponentpublicclassMyPublisherimplementsApplicationEventPublisherAware,ApplicationContextAware{privateApplicationEventPublisher applicationEventPublisher;privateApplicationContext applicationContext;@OverridepublicvoidsetApplicationEventPublisher(ApplicationEventPublisher applicationEventPublisher){this.applicationEventPublisher = applicationEventPublisher;}@OverridepublicvoidsetApplicationContext(ApplicationContext applicationContext)throwsBeansException{this.applicationContext = applicationContext;}//发送广播消息publicvoidpublishEvent(){System.out.println("我要开始发送消息了。。。");MyEvent myEvent =newMyEvent(applicationContext);
        applicationEventPublisher.publishEvent(myEvent);}}
MyPublisher

实现了

ApplicationEventPublisherAware

接口 ,在

spring

初始化(见上文中的

invokeAwareInterfaces

)的时候会回调

setApplicationEventPublisher

方法,获取到初始化(添加

bean

后置处理器

ApplicationContextAwareProcessor

)时的

AbstractApplicationContext

,而

AbstractApplicationContext

又间接实现了

ApplicationEventPublisher

而获得发送能力。真正执行的是

AbstractApplicationContext

类中的

publishEvent

方法。

自定义监听

@ComponentpublicclassMyEventListenerimplementsApplicationListener<MyEvent>{@OverridepublicvoidonApplicationEvent(MyEvent event){System.out.println("我监听到你的消息了");}}
MyEventListener

实现了

ApplicationListener

接口,在

spring

初始化(见上文中的

addApplicationListener

)的时候会添加到

applicationListeners

中,在执行

publishEvent

方法时就会走

MyEventListener

中的

onApplicationEvent

方法。

客户端

@RestController@RequestMapping("/demo")publicclassDemoTest{@AutowiredprivateMyPublisher myPublisher;@RequestMapping("/test")publicvoidtest(){
        myPublisher.publishEvent();}}

访问

127.0.0.1:8008/demo/test

就可以发送广播了,发送与监听内容如下:

我要开始发送消息了。。。
我监听到你的消息了

看到这儿,相信你己经完全掌握了广播与监听的精髓了,赶快实践起来吧。

看完文章,再看看我为了写这篇文章又日渐稀少的头发,我忍不住哭出声来。可能只有给我点赞,才能平复我的心情吧。

好看的皮囊千篇一律,有趣的灵魂万里挑一,让我们在冷漠的城市里相互温暖,我是阿Q,我们下期再见!

标签: spring 数据库 mysql

本文转载自: https://blog.csdn.net/Qingai521/article/details/131548884
版权归原作者 阿Q说代码 所有, 如有侵权,请联系我们删除。

“【Spring】从Spring源码入手分析广播与监听并完成项目实战”的评论:

还没有评论