likes
comments
collection
share

应用启动过程——SpringApplicationRunListener

作者站长头像
站长
· 阅读数 47

应用启动过程——SpringApplicationRunListener

在前一节中,我们已经隐隐约约瞥到过SpringApplicationRunListener的身影,这一节就来详细读一下它的相关源码并尝试使用。

1 SpringApplicationRunListener是什么

1.1 读注释

/**
 * Listener for the {@link SpringApplication} {@code run} method.
 * {@link SpringApplicationRunListener}s are loaded through the
 * {@link SpringFactoriesLoader} and should declare a public constructor that accepts a
 * {@link SpringApplication} instance and a {@code String[]} of arguments. A new
 * {@link SpringApplicationRunListener} instance will be created for each run.
 */
public interface SpringApplicationRunListener {

翻译过来就是:

  • 监听的是SpringApplication的run方法——于是我们可以很容易记住它的名字“SpringApplicationRunListener”,而且知道了SpringApplicationRunListener只会在run方法的执行过程中被调用
  • 监听类是通过SpringFactoriesLoader加载的——这点稍后可以找到代码印证一下,应该可以找到注册Listener的方式
  • 监听类应当声明一个入参为(SpringApplication, String[])的构造方法
  • 每次运行都会新生成一个Listener实例——在使用SpringApplicationRunListener的时候要注意这点

1.2 读接口方法

应用启动过程——SpringApplicationRunListener

大概翻看一下SpringApplicationRunListener类,就可以知道,主要是定义了一堆会在不同时机被调用的方法。

1.3 SpringApplicationRunListener的加载方式

SpringApplicationRunListener的获取方式也很简单,从getRunListeners -> getSpringFactoriesInstances容易看出是通过spring.factories配置SpringApplicationRunListener

private SpringApplicationRunListeners getRunListeners(String[] args) {
   Class<?>[] types = new Class<?>[] { SpringApplication.class, String[].class };
   return new SpringApplicationRunListeners(logger,
         getSpringFactoriesInstances(SpringApplicationRunListener.class, types, this, args),
         this.applicationStartup);
}
private <T> Collection<T> getSpringFactoriesInstances(Class<T> type, Class<?>[] parameterTypes, Object... args) {
	 ClassLoader classLoader = getClassLoader();
	 // Use names and ensure unique to protect against duplicates
	 Set<String> names = new LinkedHashSet<>(SpringFactoriesLoader.loadFactoryNames(type, classLoader));
	 List<T> instances = createSpringFactoriesInstances(type, parameterTypes, classLoader, args, names);
	 AnnotationAwareOrderComparator.sort(instances);
	 return instances;
}

单纯看SpringApplicationRunListener的确很简单,也就是以上这些内容。为了能继续深入学习,举一反三,我们debug一下getRunListeners的代码,可以看到SpringBoot在启动时有一个EventPublishingRunListener,接下来以它为入口,看看还能多学到点什么。

应用启动过程——SpringApplicationRunListener

2 EventPublishingRunListener

我们围绕EventPublishingRunListener来分析一下

2.1 EventPublishingRunListener跟读

看注释及类继承关系可知,EventPublishingRunListener既是一个监听者(它实现了SpringApplicationRunListener),又是一个事件发布者(它在监听方法中,发布了相应的事件)。

/**
 * {@link SpringApplicationRunListener} to publish {@link SpringApplicationEvent}s.
 * <p>
 * Uses an internal {@link ApplicationEventMulticaster} for the events that are fired
 * before the context is actually refreshed.
 *
 * @author Phillip Webb
 * @author Stephane Nicoll
 * @author Andy Wilkinson
 * @author Artsiom Yudovin
 * @author Brian Clozel
 * @author Chris Bono
 * @since 1.0.0
 */
public class EventPublishingRunListener implements SpringApplicationRunListener, Ordered {

再看一下EventPublishingRunListener实现的监听方法,以starting为例,发现它是通过持有的SimpleApplicationEventMulticaster发布事件出去的

@Override
public void starting(ConfigurableBootstrapContext bootstrapContext) {
   this.initialMulticaster
      .multicastEvent(new ApplicationStartingEvent(bootstrapContext, this.application, this.args));
}

那事件的监听者又是从哪里来的呢,继续跟进代码multicastEvent,方法内调用了getApplicationListeners获取事件监听者,并逐个调用

public void multicastEvent(final ApplicationEvent event, @Nullable ResolvableType eventType) {
   ResolvableType type = (eventType != null ? eventType : resolveDefaultEventType(event));
   Executor executor = getTaskExecutor();
   for (ApplicationListener<?> listener : getApplicationListeners(event, type)) {
      if (executor != null) {
         executor.execute(() -> invokeListener(listener, event));
      }
      else {
         invokeListener(listener, event);
      }
   }
}

跟进getApplicationListeners -> retrieveApplicationListeners,发现了事件监听者的来源:this.defaultRetriever

private Collection<ApplicationListener<?>> retrieveApplicationListeners(
      ResolvableType eventType, @Nullable Class<?> sourceType, @Nullable CachedListenerRetriever retriever) {

   List<ApplicationListener<?>> allListeners = new ArrayList<>();
   Set<ApplicationListener<?>> filteredListeners = (retriever != null ? new LinkedHashSet<>() : null);
   Set<String> filteredListenerBeans = (retriever != null ? new LinkedHashSet<>() : null);

   Set<ApplicationListener<?>> listeners;
   Set<String> listenerBeans;
   synchronized (this.defaultRetriever) {
      listeners = new LinkedHashSet<>(this.defaultRetriever.applicationListeners);
      listenerBeans = new LinkedHashSet<>(this.defaultRetriever.applicationListenerBeans);
   }

   //省略余下代码...
   //... ...
}

找到this.defaultRetriever这个属性赋值的地方,果然有个addApplicationListener

@Override
public void addApplicationListener(ApplicationListener<?> listener) {
   synchronized (this.defaultRetriever) {
      // Explicitly remove target for a proxy, if registered already,
      // in order to avoid double invocations of the same listener.
      Object singletonTarget = AopProxyUtils.getSingletonTarget(listener);
      if (singletonTarget instanceof ApplicationListener) {
         this.defaultRetriever.applicationListeners.remove(singletonTarget);
      }
      this.defaultRetriever.applicationListeners.add(listener);
      this.retrieverCache.clear();
   }
}

查找这个方法的调用方,或者直接回到EventPublishingRunListener中,很容易找到EventPublishingRunListener的构造方法中初始化添加了监听者,而监听者的源头又来自于application.getListeners()

public EventPublishingRunListener(SpringApplication application, String[] args) {
   this.application = application;
   this.args = args;
   this.initialMulticaster = new SimpleApplicationEventMulticaster();
   for (ApplicationListener<?> listener : application.getListeners()) {
      this.initialMulticaster.addApplicationListener(listener);
   }
}

再到SpringApplication类中找到listeners来源于构造方法,从spring.factories中获取

public SpringApplication(ResourceLoader resourceLoader, Class<?>... primarySources) {
   this.resourceLoader = resourceLoader;
   Assert.notNull(primarySources, "PrimarySources must not be null");
   this.primarySources = new LinkedHashSet<>(Arrays.asList(primarySources));
   this.webApplicationType = WebApplicationType.deduceFromClasspath();
   this.bootstrapRegistryInitializers = new ArrayList<>(
         getSpringFactoriesInstances(BootstrapRegistryInitializer.class));
   setInitializers((Collection) getSpringFactoriesInstances(ApplicationContextInitializer.class));
   setListeners((Collection) getSpringFactoriesInstances(ApplicationListener.class));
   this.mainApplicationClass = deduceMainApplicationClass();
}

看到这里,应该已经晕了,我们阶段性总结一下:EventPublishingRunListener本身是一个监听者,监听了SpringApplication#run启动过程的生命周期方法。EventPublishingRunListener同时还是一个事件发布者,通过ApplicationEventMulticaster事件广播器发布了各种SpringApplicationEvent事件,这些事件有自己的监听者,事件监听者最初取自于SpringApplication(构造时从spring.factories中加载)。

到此本以为一切都结束了,可是细心的我又发现一个可疑的点,请看EventPublishingRunListener部分方法的对比,以starting和started为例,为啥使用了不同的事件发布方式?context.publishEvent和AvailabilityChangeEvent.publish是什么?

public void starting(ConfigurableBootstrapContext bootstrapContext) {
	this.initialMulticaster
		.multicastEvent(new ApplicationStartingEvent(bootstrapContext, this.application, this.args));
}
public void started(ConfigurableApplicationContext context, Duration timeTaken) {
   context.publishEvent(new ApplicationStartedEvent(this.application, this.args, context, timeTaken));
   AvailabilityChangeEvent.publish(context, LivenessState.CORRECT);
}

深入代码,不难发现原来ApplicationContext继承了ApplicationEventPublisher,也是一个事件发布者。而且发布事件的机制和EventPublishingRunListener类似,也是持有一个ApplicationEventMulticaster,通过事件广播器广播事件及管理监听者,而这些监听者就是开发时通常用到的,利用ApplicationContext的addApplicationListener方法注册进去的。

2.2 EventPublishingRunListener总结

对以上内容,我总结了一份关系图供参考,从中可以整体上感受到:

  • SpringApplicationRunListener是SpringApplication#run方法执行过程的监听者,SpringApplication#run通过SpringApplicationRunListeners代理调用SpringApplicationRunListener
  • EventPublishingRunListener实现了SpringApplicationRunListener,监听SpringApplication#run方法的执行过程,并依此发布不同的事件SpringApplicationEvent
  • EventPublishingRunListener发布事件及调用监听者的机制是利用ApplicationEventMulticaster事件广播器实现的
  • ApplicationContext也有事件发布者的身份,事件发布机制与EventPublishingRunListener的类似

应用启动过程——SpringApplicationRunListener

转载自:https://juejin.cn/post/7229269625640714301
评论
请登录