springboot自動配置原理以及spring.factories文件的作用詳解
一、springboot 自動配置原理
先說說我們自己的應(yīng)用程序中Bean加入容器的辦法:
package com.ynunicom.dc.dingdingcontractapp;
import com.alibaba.druid.spring.boot.autoconfigure.DruidDataSourceAutoConfigure;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
/**
* @author jinye.Bai
*/
@SpringBootApplication(
scanBasePackages ={"com.ynunicom.dc.dingdingcontractapp"}
)
public class DingdingContractAppApplication {
public static void main(String[] args) {
SpringApplication.run(DingdingContractAppApplication.class, args);
}
}
我們在應(yīng)用程序的入口設(shè)置了 @SpringBootApplication標(biāo)簽,默認(rèn)情況下他會掃描所有次級目錄。
如果增加了 scanBasePackages屬性,就會掃描所有被指定的路徑及其次級目錄。
那么它在掃描的是什么東西呢?
是這個:@Component
所有被掃描到的 @Component,都會成為一個默認(rèn)的singleton(單例,即一個容器里只有一個對象實(shí)體)加入到容器中。
認(rèn)識到以上這點(diǎn),便于我們理解springboot自動配置的機(jī)制。
接下來讓我們看看在自己的應(yīng)用程序中實(shí)現(xiàn)配置的方法。
如圖:
package com.ynunicom.dc.dingdingcontractapp.configuration;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.client.RestTemplate;
/**
* @author: jinye.Bai
* @date: 2020/5/22 15:51
*/
@Configuration
public class RestTemplateConfig {
@Bean
public RestTemplate restTemplate(){
return new RestTemplate();
}
}
這里我們設(shè)置了一個配置,往容器中加入了一個RestTemplate。
首先說 @Configuration,這個標(biāo)簽繼承了 @Component標(biāo)簽,我們可以在標(biāo)簽內(nèi)容看到:
//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by Fernflower decompiler)
//
package org.springframework.context.annotation;
import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
import org.springframework.core.annotation.AliasFor;
import org.springframework.stereotype.Component;
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Component
public @interface Configuration {
@AliasFor(
annotation = Component.class
)
String value() default "";
}
可以看到其中是有 @Component標(biāo)簽的,所以,@Configuration會被 @SpringBootApplication掃描到,進(jìn)而把它和它下面的 @Bean加入容器,于是我們 RestTemplate的內(nèi)容就配置完成了,在后續(xù)的使用中,我們就可以直接從容器中拿出RestTemplate使用它。
對于在maven中引用的其他外部包加入容器的過程,需要用到spring.factories。
二、spring.factories文件的作用
在springboot運(yùn)行時(shí),SpringFactoriesLoader 類會去尋找
public static final String FACTORIES_RESOURCE_LOCATION = "META-INF/spring.factories";
我們以mybatis-plus為例。
首先我們引入:
<dependency>
<groupId>com.baomidou</groupId>
<artifactId>mybatis-plus-boot-starter</artifactId>
<version>3.3.2</version>
</dependency>
然后去maven的依賴?yán)锟此淖詣优渲妙怣ybatisPlusAutoConfiguration
@Configuration
@ConditionalOnClass({SqlSessionFactory.class, SqlSessionFactoryBean.class})
@ConditionalOnSingleCandidate(DataSource.class)
@EnableConfigurationProperties({MybatisPlusProperties.class})
@AutoConfigureAfter({DataSourceAutoConfiguration.class, MybatisPlusLanguageDriverAutoConfiguration.class})
public class MybatisPlusAutoConfiguration implements InitializingBean {
可以看到有上文提到的 @Configuration,還有從application.yml載入自動配置的 @EnableConfigurationProperties({MybatisPlusProperties.class})
這個注解的具體內(nèi)容請查看我另一篇博文,對其進(jìn)行了解釋:
迅速學(xué)會@ConfigurationProperties的使用
也就是說,springboot只要能掃描到MybatisPlusAutoConfiguration類的 @Configuration注解,其中的所有配置就能自動加入到容器中,這一過程由上面提到的SpringFactoriesLoader 起作用,它會去尋找 “META-INF/spring.factories” 文件,我們可以在 mybatis-plus的依賴中找到它:

SpringFactoriesLoader為什么要讀取它呢?因?yàn)樗鼉?nèi)部是這樣的
org.springframework.boot.autoconfigure.EnableAutoConfiguration=\ com.baomidou.mybatisplus.autoconfigure.MybatisPlusLanguageDriverAutoConfiguration,\ com.baomidou.mybatisplus.autoconfigure.MybatisPlusAutoConfiguration
spring.factories用鍵值對的方式記錄了所有需要加入容器的類,EnableAutoConfigurationImportSelector的selectImports方法返回的類名,來自spring.factories文件內(nèi)的配置信息,這些配置信息的key等于EnableAutoConfiguration,因?yàn)閟pring boot應(yīng)用啟動時(shí)使用了EnableAutoConfiguration注解,所以EnableAutoConfiguration注解通過import注解將EnableAutoConfigurationImportSelector類實(shí)例化,并且將其selectImports方法返回的類名實(shí)例化后注冊到spring容器。
以上內(nèi)容是springboot獲得這些類的方式,如果你想要實(shí)現(xiàn)自己的自動配置,就將你的類通過鍵值對的方式寫在你的spring.factories即可,注意,值是你的自動配置類,鍵必須是org.springframework.boot.autoconfigure.EnableAutoConfiguration
spring.factories 的妙用
現(xiàn)象
在閱讀 Spring-Boot 相關(guān)源碼時(shí),常常見到 spring.factories 文件,里面寫了自動配置(AutoConfiguration)相關(guān)的類名,因此產(chǎn)生了一個疑問:“明明自動配置的類已經(jīng)打上了 @Configuration 的注解,為什么還要寫 spring.factories 文件?
用過 Spring Boot 的都知道
@ComponentScan 注解的作用是掃描 @SpringBootApplication 所在的 Application 類所在的包(basepackage)下所有的 @component 注解(或拓展了 @component 的注解)標(biāo)記的 bean,并注冊到 spring 容器中。
那么問題來了
在 Spring Boot 項(xiàng)目中,如果你想要被 Spring 容器管理的 bean 不在 Spring Boot 包掃描路徑下,怎么辦?
解決 Spring Boot 中不能被默認(rèn)路徑掃描的配置類的方式,有 2 種:
(1)在 Spring Boot 主類上使用 @Import 注解
(2)使用 spring.factories 文件
以下是對 使用 spring.factories 文件的簡單理解
Spring Boot 的擴(kuò)展機(jī)制之 Spring Factories
Spring Boot 中有一種非常解耦的擴(kuò)展機(jī)制:Spring Factories。這種擴(kuò)展機(jī)制實(shí)際上是仿照J(rèn)ava中的SPI擴(kuò)展機(jī)制來實(shí)現(xiàn)的。
什么是 SPI 機(jī)制?
SPI 的全名為 Service Provider Interface.大多數(shù)開發(fā)人員可能不熟悉,因?yàn)檫@個是針對廠商或者插件的。在java.util.ServiceLoader的文檔里有比較詳細(xì)的介紹。
簡單的總結(jié)下 java SPI 機(jī)制的思想。我們系統(tǒng)里抽象的各個模塊,往往有很多不同的實(shí)現(xiàn)方案,比如日志模塊的方案,xml解析模塊、jdbc模塊的方案等。面向的對象的設(shè)計(jì)里,我們一般推薦模塊之間基于接口編程,模塊之間不對實(shí)現(xiàn)類進(jìn)行硬編碼。一旦代碼里涉及具體的實(shí)現(xiàn)類,就違反了可拔插的原則,如果需要替換一種實(shí)現(xiàn),就需要修改代碼。為了實(shí)現(xiàn)在模塊裝配的時(shí)候能不在程序里動態(tài)指明,這就需要一種服務(wù)發(fā)現(xiàn)機(jī)制。
java SPI 就是提供這樣的一個機(jī)制:為某個接口尋找服務(wù)實(shí)現(xiàn)的機(jī)制。有點(diǎn)類似IOC的思想,就是將裝配的控制權(quán)移到程序之外,在模塊化設(shè)計(jì)中這個機(jī)制尤其重要。
Spring Boot 中的 SPI 機(jī)制
在 Spring 中也有一種類似與 Java SPI 的加載機(jī)制。它在 resources/META-INF/spring.factories 文件中配置接口的實(shí)現(xiàn)類名稱,然后在程序中讀取這些配置文件并實(shí)例化。
在 Spring 中也有一種類似與 Java SPI 的加載機(jī)制。它在 resources/META-INF/spring.factories 文件中配置接口的實(shí)現(xiàn)類名稱,然后在程序中讀取這些配置文件并實(shí)例化。
這種自定義的SPI機(jī)制是 Spring Boot Starter 實(shí)現(xiàn)的基礎(chǔ)。

Spring Factories 實(shí)現(xiàn)原理是什么?
spring-core 包里定義了 SpringFactoriesLoader 類,這個類實(shí)現(xiàn)了檢索 META-INF/spring.factories 文件,并獲取指定接口的配置的功能。在這個類中定義了兩個對外的方法:
loadFactories 根據(jù)接口類獲取其實(shí)現(xiàn)類的實(shí)例,這個方法返回的是對象列表。 loadFactoryNames 根據(jù)接口獲取其接口類的名稱,這個方法返回的是類名的列表。
上面的兩個方法的關(guān)鍵都是從指定的 ClassLoader 中獲取 spring.factories 文件,并解析得到類名列表,具體代碼如下
public final class SpringFactoriesLoader {
public static final String FACTORIES_RESOURCE_LOCATION = "META-INF/spring.factories";
private static final Log logger = LogFactory.getLog(SpringFactoriesLoader.class);
private static final Map<ClassLoader, MultiValueMap<String, String>> cache = new ConcurrentReferenceHashMap();
private SpringFactoriesLoader() {}
public static <T> List<T> loadFactories(Class<T> factoryClass, @Nullable ClassLoader classLoader) {
Assert.notNull(factoryClass, "'factoryClass' must not be null");
ClassLoader classLoaderToUse = classLoader;
if (classLoader == null) {
classLoaderToUse = SpringFactoriesLoader.class.getClassLoader();
}
List<String> factoryNames = loadFactoryNames(factoryClass, classLoaderToUse);
if (logger.isTraceEnabled()) {
logger.trace("Loaded [" + factoryClass.getName() + "] names: " + factoryNames);
}
List<T> result = new ArrayList(factoryNames.size());
Iterator var5 = factoryNames.iterator();
while(var5.hasNext()) {
String factoryName = (String)var5.next();
result.add(instantiateFactory(factoryName, factoryClass, classLoaderToUse));
}
AnnotationAwareOrderComparator.sort(result);
return result;
}
public static List<String> loadFactoryNames(Class<?> factoryClass, @Nullable ClassLoader classLoader) {
String factoryClassName = factoryClass.getName();
return (List)loadSpringFactories(classLoader).getOrDefault(factoryClassName, Collections.emptyList());
}
private static Map<String, List<String>> loadSpringFactories(@Nullable ClassLoader classLoader) {
MultiValueMap<String, String> result = (MultiValueMap)cache.get(classLoader);
if (result != null) {
return result;
} else {
try {
Enumeration<URL> urls = classLoader != null ? classLoader.getResources("META-INF/spring.factories") : ClassLoader.getSystemResources("META-INF/spring.factories");
LinkedMultiValueMap result = new LinkedMultiValueMap();
while(urls.hasMoreElements()) {
URL url = (URL)urls.nextElement();
UrlResource resource = new UrlResource(url);
Properties properties = PropertiesLoaderUtils.loadProperties(resource);
Iterator var6 = properties.entrySet().iterator();
while(var6.hasNext()) {
Entry<?, ?> entry = (Entry)var6.next();
String factoryClassName = ((String)entry.getKey()).trim();
String[] var9 = StringUtils.commaDelimitedListToStringArray((String)entry.getValue());
int var10 = var9.length;
for(int var11 = 0; var11 < var10; ++var11) {
String factoryName = var9[var11];
result.add(factoryClassName, factoryName.trim());
}
}
}
cache.put(classLoader, result);
return result;
} catch (IOException var13) {
throw new IllegalArgumentException("Unable to load factories from location [META-INF/spring.factories]", var13);
}
}
}
private static <T> T instantiateFactory(String instanceClassName, Class<T> factoryClass, ClassLoader classLoader) {
try {
Class<?> instanceClass = ClassUtils.forName(instanceClassName, classLoader);
if (!factoryClass.isAssignableFrom(instanceClass)) {
throw new IllegalArgumentException("Class [" + instanceClassName + "] is not assignable to [" + factoryClass.getName() + "]");
} else {
return ReflectionUtils.accessibleConstructor(instanceClass, new Class[0]).newInstance();
}
} catch (Throwable var4) {
throw new IllegalArgumentException("Unable to instantiate factory class: " + factoryClass.getName(), var4);
}
}
}
從代碼中我們可以知道,在這個方法中會遍歷整個 spring-boot 項(xiàng)目的 classpath 下 ClassLoader 中所有 jar 包下的 spring.factories文件。也就是說我們可以在自己的 jar 中配置 spring.factories 文件,不會影響到其它地方的配置,也不會被別人的配置覆蓋。
Spring Factories 在 Spring Boot 中的應(yīng)用
在 Spring Boot 的很多包中都能夠找到 spring.factories 文件,接下來我們以 spring-boot-autoconfigure 包為例進(jìn)行介紹
# Initializers org.springframework.context.ApplicationContextInitializer=\ org.springframework.boot.autoconfigure.SharedMetadataReaderFactoryContextInitializer,\ org.springframework.boot.autoconfigure.logging.ConditionEvaluationReportLoggingListener # Application Listeners org.springframework.context.ApplicationListener=\ org.springframework.boot.autoconfigure.BackgroundPreinitializer # Auto Configuration Import Listeners org.springframework.boot.autoconfigure.AutoConfigurationImportListener=\ org.springframework.boot.autoconfigure.condition.ConditionEvaluationReportAutoConfigurationImportListener # Auto Configuration Import Filters org.springframework.boot.autoconfigure.AutoConfigurationImportFilter=\ org.springframework.boot.autoconfigure.condition.OnBeanCondition,\ org.springframework.boot.autoconfigure.condition.OnClassCondition,\ org.springframework.boot.autoconfigure.condition.OnWebApplicationCondition # Auto Configure org.springframework.boot.autoconfigure.EnableAutoConfiguration=\ org.springframework.boot.autoconfigure.admin.SpringApplicationAdminJmxAutoConfiguration,\ org.springframework.boot.autoconfigure.aop.AopAutoConfiguration,\ org.springframework.boot.autoconfigure.amqp.RabbitAutoConfiguration,\ org.springframework.boot.autoconfigure.batch.BatchAutoConfiguration,\ org.springframework.boot.autoconfigure.cache.CacheAutoConfiguration
結(jié)合前面的內(nèi)容,可以看出 spring.factories 文件可以將 spring-boot 項(xiàng)目包以外的 bean(即在 pom 文件中添加依賴中的 bean)注冊到 spring-boot 項(xiàng)目的 spring 容器。
由于@ComponentScan 注解只能掃描 spring-boot 項(xiàng)目包內(nèi)的 bean 并注冊到 spring 容器中,因此需要 @EnableAutoConfiguration 注解來注冊項(xiàng)目包外的bean。
而 spring.factories 文件,則是用來記錄項(xiàng)目包外需要注冊的bean類名。
以上為個人經(jīng)驗(yàn),希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
SpringBoot與SpringMVC中參數(shù)傳遞的原理解析
這篇文章主要介紹了SpringBoot與SpringMVC中參數(shù)傳遞的原理,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-07-07
Java Swing JTextArea文本區(qū)域的實(shí)現(xiàn)示例
這篇文章主要介紹了Java Swing JTextArea文本區(qū)域的實(shí)現(xiàn)示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-12-12
SpringBoot配置文件中數(shù)據(jù)庫密碼加密兩種方案(推薦)
SpringBoot項(xiàng)目經(jīng)常將連接數(shù)據(jù)庫的密碼明文放在配置文件里,安全性就比較低一些,尤其在一些企業(yè)對安全性要求很高,因此我們就考慮如何對密碼進(jìn)行加密,文中給大家介紹加密的兩種方式,感興趣的朋友一起看看吧2019-10-10
@RequestParam 接收參數(shù)的值為null的處理方式
這篇文章主要介紹了@RequestParam 接收參數(shù)的值為null的處理方式,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-11-11
解決使用redisTemplate高并發(fā)下連接池滿的問題
這篇文章主要介紹了解決使用redisTemplate高并發(fā)下連接池滿的問題,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-12-12

