前言
最近在学习SpringBoot源码的时候发现了ApplicationListener和SpringApplicationRunListener两个监听器。一开始很容易将两个监听器混淆。以为都是SpringBoot的监听器。后面才发现他们是在不同的阶段被SpringBoot扫描并实例化的。
ApplicationListener加载详解
首先ApplicationListener是在SpringApplication类实例化的时候就加载的,Spring会从所有的spring.factories文件中加载所有的SpringApplication类的实现类(默认有11个)并实例化,设置为内部变量。
getSpringFactoriesInstances方法实际上是调用loadFactoryNames(type, classLoader)方法获取META-INF文件中spring.factories的key为参数type对应的所有路径。而刚传入的key就是SpringApplication。
spring.factories文件
加载到的ApplicationListener列表
SpringApplicationRunListener加载详解
SpringApplicationRunListener是在实例化完成后调用run方法中获取的。getRunListeners(args)方法中同样是调用getSpringFactoriesInstances方法,不过传入的参数是org.springframework.boot.SpringApplicationRunListener,去META-INF中spring.factories中获取key为org.springframework.boot.SpringApplicationRunListener的SpringApplicationRunListener自动配置类。获取到EventPublishingRunListener类(时间发布监听器),监听启动过程中不同的生命周期。
加载完所有的SpringApplicationRunListener类后会调用starting方法,starting中调用EventPublishingRunListener的静态内部类initialMulticaster的multicastEvent方法,参数是SpringApplication。再在该方法中拿到参数SpringApplication中的监听器,也就是上面实例化过程中的所有ApplicationListener。在通过反射代理执行这些监听器。
区别
1.首先是所在包不同,ApplicationListener是spring上下文org.springframework.context中的监听器,作用于整个Spring上下文,范围更广。SpringApplicationRunListener则是org.springframework.boot包下的,监听的是SpringBoot程序。
2.ApplicationListener是在实例化阶段获取的,SpringApplicationRunListener是在初始化SpringBoot应用时获取的。ApplicationListener的实现类会被SpringApplicationRunListener代理执行。