Springboot 系列(三)Spring Boot 自動配置原理

来源:https://www.cnblogs.com/niumoo/archive/2019/02/18/10393547.html
-Advertisement-
Play Games

註意:本 Spring Boot 系列文章基於 Spring Boot 版本 v2.1.1.RELEASE 進行學習分析,版本不同可能會有細微差別。 前言 關於配置文件可以配置的內容,在 "Spring Boot 官方網站" 已經提供了完整了配置示例和解釋。 可以這麼說,Spring Boot 的一 ...


註意:本 Spring Boot 系列文章基於 Spring Boot 版本 v2.1.1.RELEASE 進行學習分析,版本不同可能會有細微差別。

前言

關於配置文件可以配置的內容,在 Spring Boot 官方網站已經提供了完整了配置示例和解釋。

可以這麼說,Spring Boot 的一大精髓就是自動配置,為開發省去了大量的配置時間,可以更快的融入業務邏輯的開發,那麼自動配置是怎麼實現的呢?

1. @SpringBootApplication

跟著 Spring Boot 的啟動類的註解 @SpringBootApplication 進行源碼跟蹤,尋找自動配置的原理。

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@SpringBootConfiguration
@EnableAutoConfiguration
@ComponentScan(
    excludeFilters = {@Filter(
    type = FilterType.CUSTOM,
    classes = {TypeExcludeFilter.class}
), @Filter(
    type = FilterType.CUSTOM,
    classes = {AutoConfigurationExcludeFilter.class}
)}
)
public @interface SpringBootApplication {

@EnableAutoConfiguration 開啟自動配置。

@ComponentScan 開啟註解掃描

SpringBootApplication 我們可以發現,這是一個簡便的註解配置,它包含了自動配置,配置類,包掃描等一系列功能。

2. @EnableAutoConfiguration

繼續跟蹤,查看@EnableAutoConfiguration 源碼,裡面比較重要的是 @Import ,導入了一個翻譯名為自動配置的選擇器的類。這個類其實就是自動配置的載入選擇器。

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@AutoConfigurationPackage
@Import({AutoConfigurationImportSelector.class})
public @interface EnableAutoConfiguration {
    String ENABLED_OVERRIDE_PROPERTY = "spring.boot.enableautoconfiguration";

    Class<?>[] exclude() default {};

    String[] excludeName() default {};
}

繼續跟蹤 AutoConfigurationImportSelector.class .在這個類有一個重要的方法 getCandidateConfigurations.用於載入 Spring Boot 配置的自動配置類。

getAutoConfigurationEntry 會篩選出有效的自動配置類。

protected AutoConfigurationEntry getAutoConfigurationEntry(
            AutoConfigurationMetadata autoConfigurationMetadata,
            AnnotationMetadata annotationMetadata) {
        if (!isEnabled(annotationMetadata)) {
            return EMPTY_ENTRY;
        }
        AnnotationAttributes attributes = getAttributes(annotationMetadata);
        List<String> configurations = getCandidateConfigurations(annotationMetadata,
                attributes);
        configurations = removeDuplicates(configurations);
        Set<String> exclusions = getExclusions(annotationMetadata, attributes);
        checkExcludedClasses(configurations, exclusions);
        configurations.removeAll(exclusions);
        configurations = filter(configurations, autoConfigurationMetadata);
        fireAutoConfigurationImportEvents(configurations, exclusions);
        return new AutoConfigurationEntry(configurations, exclusions);
    }   

protected List<String> getCandidateConfigurations(AnnotationMetadata metadata,
            AnnotationAttributes attributes) {
        List<String> configurations = SpringFactoriesLoader.loadFactoryNames(
                getSpringFactoriesLoaderFactoryClass(), getBeanClassLoader());
        Assert.notEmpty(configurations,
                "No auto configuration classes found in META-INF/spring.factories. If you "
                        + "are using a custom packaging, make sure that file is correct.");
        return configurations;
    }

下圖是 DEBUG 模式下篩選之後的結果,因為我只添加了 web 模塊,所以只有 web 相關的自動配置。

篩選過後的自動配置

3. xxxAutoConfiguration 與 xxxProperties

在上面的 debug 里,我們看到了成功載入的自動配置,目前只看到了配置類,卻還沒有發現自動配置值,隨便選擇一個 AutoConfiguration 查看源碼。

這裡選擇了 ServletWebServerFactoryAutoConfiguration.

@Configuration
@AutoConfigureOrder(Ordered.HIGHEST_PRECEDENCE)
//判斷當前項目有沒有這個類
//CharacterEncodingFilter;SpringMVC中進行亂碼解決的過濾器;
@ConditionalOnClass(ServletRequest.class)
//Spring底層@Conditional註解(Spring註解版),根據不同的條件,如果
//滿足指定的條件,整個配置類裡面的配置就會生效; 判斷當前應用是否是web應用,如果是,當前配置類生效
@ConditionalOnWebApplication(type = Type.SERVLET)
@EnableConfigurationProperties(ServerProperties.class)
@Import({ ServletWebServerFactoryAutoConfiguration.BeanPostProcessorsRegistrar.class,
        ServletWebServerFactoryConfiguration.EmbeddedTomcat.class,
        ServletWebServerFactoryConfiguration.EmbeddedJetty.class,
        ServletWebServerFactoryConfiguration.EmbeddedUndertow.class })
public class ServletWebServerFactoryAutoConfiguration {

需要註意的是 @EnableConfigurationProperties(ServerProperties.class).他的意思是啟動指定類的
ConfigurationProperties功能;將配置文件中對應的值和 ServerProperties 綁定起來;並把
ServerProperties 加入到 IOC 容器中。

再來看一下 ServerProperties .

@ConfigurationProperties(prefix = "server", ignoreUnknownFields = true)
public class ServerProperties {

    /**
     * Server HTTP port.
     */
    private Integer port;

顯而易見了,這裡使用 ConfigurationProperties 綁定屬性映射文件中的 server 開頭的屬性。結合預設配置

# 路徑spring-boot-autoconfigure-2.1.1.RELEASE.jar
# /META-INF/spring-configuration-metadata.json

    {
      "name": "server.port",
      "type": "java.lang.Integer",
      "description": "Server HTTP port.",
      "sourceType": "org.springframework.boot.autoconfigure.web.ServerProperties",
      "defaultValue": 8080
    }

達到了自動配置的目的。

4. 自動配置總結

  1. SpringBoot 啟動的時候載入主配置類,開啟了自動配置功能 @EnableAutoConfiguration 。
  2. @EnableAutoConfiguration 給容器導入META-INF/spring.factories 里定義的自動配置類。
  3. 篩選有效的自動配置類。
  4. 每一個自動配置類結合對應的 xxxProperties.java 讀取配置文件進行自動配置功能 。

5. 配置類

通過自動配置,我們發現已經幫我們省去了大量的配置文件的編寫,那麼在自定義配置的時候,我們是不是需要編寫XML呢?Spring boot 儘管可以使用 SpringApplicationXML 文件進行配置,但是我們通常會使用 @Configuration 類進行代替,這也是官方推薦的方式。

5.1 XML配置

定義 helloService Bean.

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">

    <bean id="helloService" class="net.codingme.boot.service.HelloService"></bean>

</beans>

引入配置。

@ImportResource(value = "classpath:spring-service.xml")
@SpringBootApplication
public class BootApplication {

    public static void main(String[] args) {
        SpringApplication.run(BootApplication.class, args);
    }
}

5.2 註解配置

此種方式和上面的XML配置是等效的,也是官方推薦的方式。@Configuration 註解的類(要在掃描的包路徑中)會被掃描到。

/**
 * <p>
 * 配置類,相當於傳統Spring 開發中的 xml-> bean的配置
 *
 * @Author niujinpeng
 * @Date 2018/12/7 0:04
 */
@Configuration
public class ServiceConfig {

    /**
     * 預設添加到容器中的 ID 為方法名(helloService)
     *
     * @return
     */
    @Bean
    public HelloService helloService() {
        return new HelloService();
    }
}

6. 附錄

@Conditional擴展註解 作用(判斷是否滿足當前指定條件)
@ConditionalOnJava 系統的java版本是否符合要求
@ConditionalOnBean 容器中存在指定Bean;
@ConditionalOnMissingBean 容器中不存在指定Bean;
@ConditionalOnExpression 滿足SpEL表達式指定
@ConditionalOnClass 系統中有指定的類
@ConditionalOnMissingClass 系統中沒有指定的類
@ConditionalOnSingleCandidate 容器中只有一個指定的Bean,或者這個Bean是首選Bean
@ConditionalOnProperty 系統中指定的屬性是否有指定的值
@ConditionalOnResource 類路徑下是否存在指定資源文件
@ConditionalOnWebApplication 當前是web環境
@ConditionalOnNotWebApplication 當前不是web環境
@ConditionalOnJndi JNDI存在指定項

文章代碼已經上傳到 GitHub Spring Boot 自動配置

<完>
本文原發於個人博客:https://www.codingme.net 轉載請註明出處


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • [TOC] vue.js 學習筆記 前言 好! MVVM 和 MVC 架構 MVVM:M 是模型(數據),V是視圖(界面),VM是V和M的雙向協調者。 MVC:M 是模型,V是視圖,C是控制器(業務) vue使用了MVVM架構來設計框架。架構是說邏輯分層,框架是指具體的實現。很明顯VM的實現是這個框 ...
  • 一 列表標簽 列表標簽分為三種。 1、無序列表<ul>,無序列表中的每一項是<li> 英文單詞解釋如下: a.ul:unordered list,“無序列表”的意思。 b.li:list item,“列表項”的意思。 示例: 1 2 3 4 5 <ul> <li>張三</li> <li>李四</li ...
  • 如果第二次看到我的文章,歡迎右側掃碼訂閱我喲~ 👉 本文長度為3633字,建議閱讀10分鐘。 堅持原創,每一篇都是用心之作~ 如果我們的開發工作真的就如搭積木一般就好了,輪廓分明,個個分開,壞了哪塊積木換掉哪塊就好了。 但是,實際我們的工作中所面臨的可能只有一塊積木,而且還是一大塊,要換得一起換, ...
  • 在大數據盛行的現在,大屏數據可視化也已經成為了一個熱門的話題。大屏可視化可以運用在眾多領域中,比如工業互聯網、醫療、交通、工業控制等等。將各項重要指標數據以圖表、各種圖形等形式表現在一個頁面上,各種數據一目瞭然。隨著瀏覽器不斷發展完善,使用 Web 做大屏展示也已經不是新鮮的事了。市面上已有不少的大... ...
  • LieBrother公眾號原文: "行為型模式:模板方法" 十一大行為型模式之一:模板方法。 簡介 姓名 :模板方法 英文名 :Template Method Pattern 價值觀 :在我的掌控下,任由你發揮 個人介紹 : Define the skeleton of an algorithm i ...
  • 題意 "題目鏈接" Sol 線段樹合併板子題 cpp include using namespace std; const int MAXN = 400000, SS = MAXN 21; inline int read() { char c = getchar(); int x = 0, f = ...
  • 《c陷阱與缺陷》 編程者也許認為,程式一旦執行上述操作完畢,就可以自由地進行讀取和寫入的操作了。遺憾的是,事實總難遂人所願,為了保持與過去不能同時進行讀寫操作的程式的向下相容性,一個輸入操作不能隨後直接緊跟輸出操作,反之亦然。如果同時進行輸入和輸出操作,必須在其中插入fseek函數調用。 這個陷阱把 ...
  • 題意 "題目鏈接" Sol 說一個尾碼自動機+線段樹的無腦做法 首先建出SAM,然後對parent樹進行dp,維護最大次大值,最小次小值 顯然一個串能更新答案的區間是$[len_{fa_{x}} + 1, len_x]$,方案數就相當於是從$siz_x$裡面選兩個,也就是$\frac{siz_x ( ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...