SpringBoot自動配置深入探究實現(xiàn)原理
說明:在閱讀本篇文章之前建議大家先詳細(xì)學(xué)習(xí)一下spring的相關(guān)知識,有助于更深刻的理解spirngboot的配置原理。
一、什么是springboot自動配置
SpringBoot通過@EnableAutoConfiguration注解開啟自動配置,對jar包下的spring.factories文件進(jìn)行掃描,這個文件中包含了可以進(jìn)行自動配置的類,當(dāng)滿足@Condition注解指定的條件時,便在依賴的支持下進(jìn)行實例化,注冊到Spring容器中。
通俗的來講,我們之前在寫ssm項目時候,配置了大量坐標(biāo)和配置內(nèi)容,搭環(huán)境的過程在項目開發(fā)中占據(jù)了大量時間,SpringBoot的最大的特點就是簡化了各種xml配置內(nèi)容,所以springboot的自動配置就是用注解來對一些常規(guī)的配置做默認(rèn)配置,簡化xml配置內(nèi)容,使你的項目能夠快速運行。
springboot核心配置原理:
- 自動配置類都存放在spring-boot-autoconfigure-版本號.jar下的org.springframework.boot.autoconfigure中
- 當(dāng)我們在application.properties中配置debug=true后啟動容器??梢钥吹椒?wù)器初始化的初始化配置
- DispatcherServletAutoConfigratio注冊前端控制器
- EmbeddedServletContainerAutoConfiguration注冊容器類型
- HttpMessageConvertersAutoConfiguration注冊json或者xml處理器
- JacksonAutoConfiguration注冊json對象解析器
- 如果加入其他功能的依賴,springBoot還會實現(xiàn)這些功能的自動配置

二、Starter組件
Starter組件是可被加載在應(yīng)用中的Maven依賴項項。只有在Maven配置中添加對應(yīng)的依賴配置,即可使用對應(yīng)的Starter組件。例如,添加spring-boot-starter-web依賴,就可以用于構(gòu)建RESTAPI服務(wù),其包含了SpringMVC和Tomcat內(nèi)嵌容器。
一個完整的Starter組件包括以下兩點:
- 提供自動配置功能的自動配置模塊
- 提供依賴關(guān)系管理崗功能的組件模塊,即封裝了組件所有功能,開箱即用。
spring-boot-starter-web依賴源碼
package org.springframework.boot.autoconfigure.web.servlet;
@Configuration
@ConditionalOnClass({ServletRequest.class})
@ConditionalOnWebApplication(
type = Type.SERVLET
)
@EnableConfigurationProperties({ServerProperties.class})
@Import({ServletWebServerFactoryAutoConfiguration.BeanPostProcessorsRegistrar.class})
public class ServletWebServerFactoryAutoConfiguration {
......
}
@三、三大注解
@SpringBootConfiguration:繼承自Configuration,支持JavaConfig的方式進(jìn)行配置。
@EnableAutoConfiguration:本文重點講解,主要用于開啟自動配置。
@ComponentScan:自動掃描組件,默認(rèn)掃描該類所在包及其子包下所有帶有指定注解的類,將它們自動裝配到bean容器中,會被自動裝配的注解包括@Controller、@Service、@Component、@Repository等。也可以指定掃描路徑。
四、@EnableAutoConfiguration
這個注解是幫助我們自動加載默認(rèn)配置的,它里面有兩個關(guān)鍵注解@AutoConfigurationPackage和@Import,我們來詳細(xì)了解@Import注解。
@Override
public String[] selectImports(AnnotationMetadata annotationMetadata) {
//檢查自動配置功能是否開啟,默認(rèn)開啟
if (!isEnabled(annotationMetadata)) {
return NO_IMPORTS;
}
//加載自動配置的元信息
AutoConfigurationMetadata autoConfigurationMetadata = AutoConfigurationMetadataLoader
.loadMetadata(this.beanClassLoader);
AnnotationAttributes attributes = getAttributes(annotationMetadata);
//獲取候選配置類
List<String> configurations = getCandidateConfigurations(annotationMetadata,
attributes);
//去掉重復(fù)的配置類
configurations = removeDuplicates(configurations);
//獲得注解中被exclude和excludeName排除的類的集合
Set<String> exclusions = getExclusions(annotationMetadata, attributes);
//檢查被排除類是否可實例化、是否被自動注冊配置所使用,不符合條件則拋出異常
checkExcludedClasses(configurations, exclusions);
//從候選配置類中去除掉被排除的類
configurations.removeAll(exclusions);
//過濾
configurations = filter(configurations, autoConfigurationMetadata);
//將配置類和排除類通過事件傳入到監(jiān)聽器中
fireAutoConfigurationImportEvents(configurations, exclusions);
//最終返回符合條件的自動配置類的全限定名數(shù)組
return StringUtils.toStringArray(configurations);@Import(AutoConfigurationImportSelector.class)注解,這里導(dǎo)入AutoConfigurationImportSelector類。這個類中有一個非常重要的方法——selectImports(),它幾乎涵蓋了組件自動裝配的所有處理邏輯,包括獲得候選配置類、配置類去重、排除不需要的配置類、過濾等,最終返回符合條件的自動配置類的全限定名數(shù)組。
五、SpringFactoriesLoader
spring-core包里定義了SpringFactoriesLoader類,這個類實現(xiàn)了檢索META-INF/spring.factories文件,并獲取指定接口的配置的功能。在這個類中定義了兩個對外的方法:
- loadFactories根據(jù)接口類獲取其實現(xiàn)類的實例,這個方法返回的是對象列表。
- loadFactoryNames根據(jù)接口獲取其接口類的名稱,這個方法返回的是類名的列表。
上面的兩個方法的關(guān)鍵都是從指定的ClassLoader中獲取spring.factories文件,并解析得到類名列表,具體代碼如下:
public static List<String> loadFactoryNames(Class<?> factoryClass, ClassLoader classLoader) {
String factoryClassName = factoryClass.getName();
try {
Enumeration<URL> urls = (classLoader != null ? classLoader.getResources(FACTORIES_RESOURCE_LOCATION) :
ClassLoader.getSystemResources(FACTORIES_RESOURCE_LOCATION));
List<String> result = new ArrayList<String>();
while (urls.hasMoreElements()) {
URL url = urls.nextElement();
Properties properties = PropertiesLoaderUtils.loadProperties(new UrlResource(url));
String factoryClassNames = properties.getProperty(factoryClassName);
result.addAll(Arrays.asList(StringUtils.commaDelimitedListToStringArray(factoryClassNames)));
}
return result;
}
catch (IOException ex) {
throw new IllegalArgumentException("Unable to load [" + factoryClass.getName() +
"] factories from location [" + FACTORIES_RESOURCE_LOCATION + "]", ex);
}
}由代碼可知,在這個方法中會遍歷整個ClassLoader中所有jar包下的spring.factories文件。也就是說我們可以在自己的jar中配置spring.factories文件,不會影響到其它地方的配置,也不會被別人的配置覆蓋。
spring.factories的是通過Properties解析得到的,所以我們在寫文件中的內(nèi)容都是安裝下面這種方式配置的:
com.xxx.interface=com.xxx.classname
最后希望大家在學(xué)習(xí)的過程中,能夠多了解實現(xiàn)各種功能的基本原理,有助于提高學(xué)習(xí)效率。
到此這篇關(guān)于SpringBoot自動配置深入探究實現(xiàn)原理的文章就介紹到這了,更多相關(guān)SpringBoot自動配置內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
MyBatis-Plus實現(xiàn)公共字段自動填充功能詳解
在開發(fā)中經(jīng)常遇到多個實體類有共同的屬性字段,這些字段屬于公共字段,也就是很多表中都有這些字段,能不能對于這些公共字段在某個地方統(tǒng)一處理,來簡化開發(fā)呢?MyBatis-Plus就提供了這一功能,本文就來為大家詳細(xì)講講2022-08-08
Java中的HashMap為什么會產(chǎn)生死循環(huán)
這篇文章主要介紹了Java中的HashMap為什么會產(chǎn)生死循環(huán),HashMap?死循環(huán)是一個比較常見、比較經(jīng)典的問題,下面文章我們就來徹底理解死循環(huán)的原因。需要的小伙伴可以參考一下2022-05-05

