Spring Boot小記
Spring啟動(dòng)容器對(duì)比
ClassPathXmlApplicationContext(基于XML)
AnnotationConfigApplicationContext(基于注解著洼,非WEB版)
AnnotationConfigEmbeddedWebApplicationContext(基于注解肺蔚,WEB版)
SpringBoot 啟動(dòng)流程
認(rèn)識(shí)ApplicationContextInitializer,SpringApplicationRunListener,Runner
1铺敌、ApplicationContextInitializer,在Spring上下文被刷新之前進(jìn)行初始化的操作蚕键。這個(gè)時(shí)候已經(jīng)創(chuàng)建了ApplicationContext 娃殖,但是沒(méi)有refresh(),ApplicationContextInitializer對(duì)ApplicationContext進(jìn)行初始話操作西篓。
2愈腾、SpringApplicationRunListener,對(duì)ApplicationContext的運(yùn)行各個(gè)時(shí)期的事件進(jìn)行廣播岂津,時(shí)事件能夠被ApplicationListener所監(jiān)聽到虱黄。
3、Runner吮成,Spring上下文后置處理 Runners可以是兩個(gè)接口的實(shí)現(xiàn)類: org.springframework.boot.ApplicationRunner org.springframework.boot.CommandLineRunner 其實(shí)沒(méi)有什么不同之處橱乱,除了接口中的run方法接受的參數(shù)類型是不一樣的以外。一個(gè)是封裝好的ApplicationArguments類型粱甫,另一個(gè)是直接的String不定長(zhǎng)數(shù)組類型泳叠。因此根據(jù)需要選擇相應(yīng)的接口實(shí)現(xiàn)即可。
啟動(dòng)流程
SpringBoot啟動(dòng)的時(shí)候魔种,不論調(diào)用什么方法析二,都會(huì)構(gòu)造一個(gè)SpringApplication的實(shí)例,然后調(diào)用這個(gè)實(shí)例的run方法节预,這樣就表示啟動(dòng)SpringBoot叶摄。
在run方法調(diào)用之前,也就是構(gòu)造SpringApplication的時(shí)候會(huì)進(jìn)行初始化的工作安拟,初始化的時(shí)候會(huì)做以下幾件事:
把參數(shù)sources設(shè)置到SpringApplication屬性中蛤吓,這個(gè)sources可以是任何類型的參數(shù)。本文的例子中這個(gè)sources就是MyApplication的class對(duì)象
判斷是否是web程序糠赦,并設(shè)置到webEnvironment這個(gè)boolean屬性中
找出所有的初始化器会傲,默認(rèn)有5個(gè)锅棕,設(shè)置到initializers屬性中
找出所有的應(yīng)用程序監(jiān)聽器,默認(rèn)有9個(gè)淌山,設(shè)置到listeners屬性中
找出運(yùn)行的主類(main class)
SpringApplication構(gòu)造完成之后調(diào)用run方法裸燎,啟動(dòng)SpringApplication,run方法執(zhí)行的時(shí)候會(huì)做以下幾件事:
構(gòu)造一個(gè)StopWatch泼疑,觀察SpringApplication的執(zhí)行
找出所有的SpringApplicationRunListener并封裝到SpringApplicationRunListeners中德绿,用于監(jiān)聽run方法的執(zhí)行。監(jiān)聽的過(guò)程中會(huì)封裝成事件并廣播出去讓初始化過(guò)程中產(chǎn)生的應(yīng)用程序監(jiān)聽器進(jìn)行監(jiān)聽
構(gòu)造Spring容器(ApplicationContext)退渗,并返回 3.1 創(chuàng)建Spring容器的判斷是否是web環(huán)境移稳,是的話構(gòu)造AnnotationConfigEmbeddedWebApplicationContext,否則構(gòu)造AnnotationConfigApplicationContext 3.2 初始化過(guò)程中產(chǎn)生的初始化器在這個(gè)時(shí)候開始工作 3.3 Spring容器的刷新(完成bean的解析会油、各種processor接口的執(zhí)行个粱、條件注解的解析等等)
從Spring容器中找出ApplicationRunner和CommandLineRunner接口的實(shí)現(xiàn)類并排序后依次執(zhí)行
SpringBoot 自動(dòng)裝載大致原理
在@SpringBootApplication標(biāo)簽中引入了EnableAutoConfigurationImportSelector,其中調(diào)用了selectImports()方法翻翩,方法中調(diào)用org.springframework.boot.autoconfigure.EnableAutoConfigurationImportSelector#getCandidateConfigurations方法都许,使用SpringFactoryLoader把META-INF文件夾中的spring.factories文件中EnableAutoConfiguration為key的文件加載了。 加載的文件全部都是java config配置文件(有默認(rèn)配置)嫂冻,利用@Conditional(Class<? extends Condition>[]) 標(biāo)簽梭稚,對(duì)相應(yīng)的bean進(jìn)行選擇性的加載。
SpringBoot 自動(dòng)裝載源碼分析
工廠加載類
比較基本而且重要的一個(gè)類絮吵,運(yùn)行加載了MATE-INF中的spring.factories 文件
@Conditional原理
@Conditional標(biāo)簽是全部Conditional相關(guān)標(biāo)簽的根源。 源碼中Conditional標(biāo)簽使用的是ConditionEvaluator來(lái)解析忱屑,如下 org.springframework.context.annotation.AnnotatedBeanDefinitionReader#registerBean(java.lang.Class<?>, java.lang.String, java.lang.Class<? extends java.lang.annotation.Annotation>...) org.springframework.context.annotation.ConditionEvaluator#shouldSkip(org.springframework.core.type.AnnotatedTypeMetadata, org.springframework.context.annotation.ConfigurationCondition.ConfigurationPhase)
加載java config原理
在初始化AnnotationConfigApplicationContext的時(shí)候蹬敲,對(duì)ConfigurationClassPostProcessor、AutowiredAnnotationBeanPostProcessor等類進(jìn)行了注冊(cè)莺戒。如下
org.springframework.context.annotation.AnnotationConfigApplicationContext#AnnotationConfigApplicationContext()
org.springframework.context.annotation.AnnotatedBeanDefinitionReader#AnnotatedBeanDefinitionReader(org.springframework.beans.factory.support.BeanDefinitionRegistry)
org.springframework.context.annotation.AnnotatedBeanDefinitionReader#AnnotatedBeanDefinitionReader(org.springframework.beans.factory.support.BeanDefinitionRegistry, org.springframework.core.env.Environment)
org.springframework.context.annotation.AnnotationConfigUtils#registerAnnotationConfigProcessors(org.springframework.beans.factory.support.BeanDefinitionRegistry)
org.springframework.context.annotation.AnnotationConfigUtils#registerAnnotationConfigProcessors(org.springframework.beans.factory.support.BeanDefinitionRegistry, java.lang.Object)
ConfigurationClassPostProcessor是一個(gè)BeanFactoryPostProcessor伴嗡,所以會(huì)對(duì)BeanDefinitionRegistry或者BeanDefinition創(chuàng)建之后進(jìn)行后置加工(refresh方法中,已經(jīng)創(chuàng)建了BeanFactory从铲,具體到運(yùn)行到哪里看源碼)瘪校。
@import() 原理
import解析原理根據(jù)的是ConfigurationClassPostProcessor,ConfigurationClassPostProcessor的加載過(guò)程參考上面 主要分析為啥@import標(biāo)簽是引入配置的但是卻能夠調(diào)用Selector的方法 org.springframework.context.annotation.ConfigurationClassPostProcessor#postProcessBeanDefinitionRegistry org.springframework.context.annotation.ConfigurationClassPostProcessor#processConfigBeanDefinitions org.springframework.context.annotation.ConfigurationClassParser#parse(java.util.Set<org.springframework.beans.factory.config.BeanDefinitionHolder>) org.springframework.context.annotation.ConfigurationClassParser#processDeferredImportSelectors 接著如下名段,調(diào)用了selectImports方法
private void processDeferredImportSelectors() {
List<DeferredImportSelectorHolder> deferredImports = this.deferredImportSelectors;
this.deferredImportSelectors = null;
Collections.sort(deferredImports, DEFERRED_IMPORT_COMPARATOR);
for (DeferredImportSelectorHolder deferredImport : deferredImports) {
ConfigurationClass configClass = deferredImport.getConfigurationClass();
try {
String[] imports = deferredImport.getImportSelector().selectImports(configClass.getMetadata());
processImports(configClass, asSourceClass(configClass), asSourceClasses(imports), false);
}
catch (BeanDefinitionStoreException ex) {
throw ex;
}
catch (Exception ex) {
throw new BeanDefinitionStoreException("Failed to process import candidates for configuration class [" +
configClass.getMetadata().getClassName() + "]", ex);
}
}
}