SpringBoot自動配置原理,你真的懂嗎?(簡單易懂)
概述
上面博文(SpringBoot簡介與快速搭建)我們簡單的介紹瞭什麼是SpringBoot,以及如何使用SpringBoot,但是我們對於SpringBoot的基本原理並沒有介紹,這篇博文我們重點介紹SpringBoot是如何實現的自動配置。
依賴管理
在我們的pom文件中最核心的依賴就一個:
<parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>2.4.4</version> <relativePath/> </parent>
它的父項目依賴,規定所有依賴的版本信息:
<parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-dependencies</artifactId> <version>2.4.4</version> </parent>
由此,我們發現springboot框架幾乎聲明瞭所有開發中常用的依賴的版本號,無需關註版本號,而且實現瞭自動版本仲裁機制,當然瞭我們也可以根據我們的需要,替換掉默認的依賴版本。
核心註解@SpringBootApplication
@SpringBootApplication public class BootApplication { public static void main(String[] args) { SpringApplication.run(BootApplication.class, args); } }
在上面的啟動類中我們發現瞭一個陌生的註解@SpringBootApplication,這個註解的是什麼含義呢?我們點進去看一下。
@SpringBootConfiguration @EnableAutoConfiguration @ComponentScan(excludeFilters = { @Filter(type = FilterType.CUSTOM, classes = TypeExcludeFilter.class), @Filter(type = FilterType.CUSTOM, classes = AutoConfigurationExcludeFilter.class) })
其實@SpringBootApplication是上面三個註解的組合體,我們對這三個註解理解清楚就可以瞭,下面逐個進行解釋:
@SpringBootConfiguration
@Configuration public @interface SpringBootConfiguration {
@Configuration我們並不陌生,它允許在上下文中註冊額外的bean或導入其他配置類,@SpringBootConfiguration其實代表當前類是一個配置類。
@EnableAutoConfiguration
EnableAutoConfiguration的目的是啟動SpringBoot的自動配置機制。
@AutoConfigurationPackage @Import(AutoConfigurationImportSelector.class) public @interface EnableAutoConfiguration {
1、AutoConfigurationPackage指定默認的包規則
@Import(AutoConfigurationPackages.Registrar.class) public @interface AutoConfigurationPackage {
AutoConfigurationPackage註解的作用是將 添加該註解的類所在的package 作為 自動配置package 進行管理。也就是說當SpringBoot應用啟動時默認會將啟動類所在的package作為自動配置的package。然後使用@Import註解將其註入到ioc容器中。這樣,可以在容器中拿到該路徑。
static class Registrar implements ImportBeanDefinitionRegistrar, DeterminableImports { @Override public void registerBeanDefinitions(AnnotationMetadata metadata, BeanDefinitionRegistry registry) { register(registry, new PackageImports(metadata).getPackageNames().toArray(new String[0])); } @Override public Set<Object> determineImports(AnnotationMetadata metadata) { return Collections.singleton(new PackageImports(metadata)); } }
重點看下registerBeanDefinitions方法。
方法的第二個參數通過new PackageImport(metadata).getPackageName()
方法設置。
接著看下PackageImport的構造器方法。
PackageImports(AnnotationMetadata metadata) { AnnotationAttributes attributes = AnnotationAttributes .fromMap(metadata.getAnnotationAttributes(AutoConfigurationPackage.class.getName(), false)); List<String> packageNames = new ArrayList<>(Arrays.asList(attributes.getStringArray("basePackages"))); for (Class<?> basePackageClass : attributes.getClassArray("basePackageClasses")) { packageNames.add(basePackageClass.getPackage().getName()); } if (packageNames.isEmpty()) { packageNames.add(ClassUtils.getPackageName(metadata.getClassName())); } this.packageNames = Collections.unmodifiableList(packageNames); }
ClassUtils.getPackageName(metadata.getClassName())獲取標註@AutoConfigurationPackage註解的類的全限定名。
最後,利用Registrar給容器中導入一系列組件,將指定的包下的所有組件導入進來。
2、@Import(AutoConfigurationImportSelector.class)
使用Import自動導入所有符合自動配置條件的Bean定義並加載到IOC容器
@Override public void process(AnnotationMetadata annotationMetadata, DeferredImportSelector deferredImportSelector) { Assert.state(deferredImportSelector instanceof AutoConfigurationImportSelector, () -> String.format("Only %s implementations are supported, got %s", AutoConfigurationImportSelector.class.getSimpleName(), deferredImportSelector.getClass().getName())); AutoConfigurationEntry autoConfigurationEntry = ((AutoConfigurationImportSelector) deferredImportSelector) .getAutoConfigurationEntry(annotationMetadata); this.autoConfigurationEntries.add(autoConfigurationEntry); for (String importClassName : autoConfigurationEntry.getConfigurations()) { this.entries.putIfAbsent(importClassName, annotationMetadata); } }
1、利用getAutoConfigurationEntry(annotationMetadata);給容器中批量導入一些組件
2、調用List configurations = getCandidateConfigurations(annotationMetadata, attributes)獲取到所有需要導入到容器中的配置類
3、利用工廠加載 Map<String, List> loadSpringFactories(@Nullable ClassLoader classLoader);得到所有的組件
4、從META-INF/spring.factories位置來加載一個文件。
默認掃描我們當前系統裡面所有META-INF/spring.factories位置的文件
spring-boot-autoconfigure-2.4.4.RELEASE.jar包裡面也有META-INF/spring.factories
文件裡面寫死瞭spring-boot一啟動就要給容器中加載的所有配置類spring-boot-autoconfigure-2.4.4.RELEASE.jar/META-INF/spring.factories,一共130個自動配置類。
130個場景的所有自動配置,會在springboot啟動的時候默認全部加載。xxxxAutoConfiguration會按照條件裝配規則(@Conditional),最終會按需配置。
小結:
SpringBoot為我們的應用程序啟用瞭三個功能:自動配置,組件掃描,以及能夠在”應用類”上定義額外的配置。
@ComponentScan
@Component
在應用程序所在的軟件包上啟用掃描,指定掃描哪些Spring註解。
ServletWebServerFactoryAutoConfiguration為例
在130個場景有我們比較熟悉兩個組件,ServletWebServerFactoryAutoConfiguration和WebMvcAutoConfiguration,我們以ServletWebServerFactoryAutoConfiguration為例,看一下SpringBoot是如何自動裝配的webServer。
在註解中我們看到瞭大量以@Conditional開頭的註解,即條件裝配,滿足Conditional指定的條件,則進行組件註入。@EnableConfigurationProperties(ServerProperties.class)+@ConfigurationProperties(prefix = “server”, ignoreUnknownFields = true),讀取我們在配置文件編寫的屬性,並把它封裝到JavaBean中,以供隨時使用。
此時我們的Tomcat容器已經以Bean的形式被註入到瞭IOC容器中。
如何禁用特定的自動配置類
如果發現應用中不需要特定自動配置類,則可以使用exclude屬性@SpringBootApplication
來禁用它們,如以下示例所示:
import org.springframework.boot.autoconfigure.*; import org.springframework.boot.autoconfigure.jdbc.*; @SpringBootApplication(exclude={DataSourceAutoConfiguration.class}) //@SpringBootApplication(excludeName = {"org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration"}) public class MyApplication { }
如果該類不在類路徑中,則可以使用excludeName
註釋的屬性,並指定完全限定的名稱(全類名字符串)。定義排除項,即可以是用哪個註釋級別也可以使用屬性來定義。
總結
- SpringBoot預先加載META-INF/spring.factories中所有的自動配置類,xxxxxAutoConfiguration
- 每個自動配置類按照條件進行生效,默認都會綁定配置文件指定的值。xxxxProperties裡面拿。xxxProperties和配置文件進行瞭綁定
- 生效的配置類就會給容器中裝配很多組件,隻要容器中有這些組件,相當於有瞭這些功能
- 定制化配置
用戶直接自己@Bean替換底層的組件
用戶根據這個組件是獲取的配置文件的什麼值,可以自行修改。
EnableAutoConfiguration —> 掃描xxxxxAutoConfiguration —> 根據條件@Conditional裝配組件 —>根據xxxxProperties加載屬性值 —-> application.properties
到此這篇關於SpringBoot自動配置原理,你真的懂嗎?的文章就介紹到這瞭,更多相關SpringBoot自動配置原理內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- SpringBoot自動配置原理分析
- Springboot @Configuration與自動配置詳解
- SpringBoot自動配置特點與原理詳細分析
- 淺談SpringBoot內嵌Tomcat的實現原理解析
- Java SpringBoot自動裝配原理詳解及源碼註釋