SpringBoot自動配置原理解析

来源:https://www.cnblogs.com/jingdongkeji/archive/2023/10/25/17786875.html
-Advertisement-
Play Games

在以前單獨使用SpringMVC Web編程框架時,我們需要單獨配置_DispatcherServlet和Tomcat,使用SpringBoot之後,我們只需要引入SpringBoot-Starter-Web就能直接開始編寫Controller等Web相關的代碼,這就是SpringBoot為們提供的... ...


1: 什麼是SpringBoot自動配置

首先介紹一下什麼是SpringBoot,SpringBoost是基於Spring框架開發出來的功能更強大的Java程式開發框架,其最主要的特點是:能使程式開發者快速搭建一套開發環境。SpringBoot能將主流的開發框架(例如SpringMVC,Dubbo,Mybatis,Redis等),做到像Maven導入Jar包一樣的簡潔快速,做到開箱即用。其中最關鍵的技術就是SpringBoot定製的各種Starter,通Maven引入Starter就能快速搭建開發環境。

2: SpringBoot Starter自動裝配案例

在以前單獨使用SpringMVC Web編程框架時,我們需要單獨配置_DispatcherServletTomcat,使用SpringBoot之後,我們只需要引入SpringBoot-Starter-Web就能直接開始編寫Controller等Web相關的代碼,這就是SpringBoot為們提供的開箱即用的便捷能力,下麵就以SpringBoot-Starter-Web_來說明SpringBoot自動配置的關鍵原理

3: SpringBoot自動裝配案例源碼解析

3.1 DispatcherServlet的自動配置原理

首先我們定位到SpringBoot自動配置的Maven依賴

     <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-autoconfigure</artifactId>
        <version>${spring-boot.version}</version>
      </dependency>


在依賴的Jar包中我們可以在_META-INF/spring.factories_中找到自動配置類:

org.springframework.boot.autoconfigure.web.servlet.DispatcherServletAutoConfiguration


在這個類中存在有一個靜態內部類:

org.springframework.boot.autoconfigure.web.servlet.DispatcherServletAutoConfiguration.DispatcherServletConfiguration


下圖是這個配置類的主要源碼和解析:

下麵將上圖中關鍵的註解功能,分別進行功能說明

3.1.1:@EnableConfigurationProperties({WebMvcProperties.class})註解解析

這個註解表示使_WebMvcProperties.class類上的@ConfigurationProperties這個註解生效,同時@ConfigurationProperties這個註解是將application.xml中以spring.mvc開頭的配置參數自動註入到WebMvcProperties.class_類的欄位中

3.1.2:@Conditional({DefaultDispatcherServletCondition.class}註解解析

該註解的原理就是將滿足特定條件情況下的Bean自動載入到Spring容器中,該註解對應的Spring介面就是_org.springframework.context.annotation.Condition_這個介面

public interface Condition {
    boolean matches(ConditionContext var1, AnnotatedTypeMetadata var2);
}


3.1.3:@ConditionOnClass註解解析

_@ConditionOnClass這個註解是在當程式代碼環境classpath下存在xxx.class的情況下條件成立,同時最終也會調用到matches_方法中,其中關鍵的源碼如下:

protected static Class<?> resolve(String className, ClassLoader classLoader) throws ClassNotFoundException {
    return classLoader != null ? Class.forName(className, false, classLoader) : Class.forName(className);
}


從上面可以看到,代碼利用_Class.forName方法載入classpath下的xxx.class類,如果載入成功條件就會成立。最後,在滿足了所有@ConditionOnal註解條件後,SpringBoot就會自動為我們在Spring容器中註入DispatcherServlet了,無需單獨配置了,直接引入spring-boot-starter-web_r即可開始使用web相關功能。

3.1.4:總結

我們以DispatcherServlet是如何自動配置到容器中為例,探究了SpringBoot Starter的自動配置原理,其中涉及了幾個關鍵的註解和步驟:

第一步:涉及到了配置文件的讀取和個性化配置,這裡就涉及到了下麵這兩個註解

@ConfigurationProperties
@EnableConfigurationProperties


第二步:設計到了在什麼條件下才自動配置的註解

@Conditional
@ConditionalOnClass


第三步:約定了自動配置類的載入路徑

/META-INF/spring-factories
org.springframework.boot.autoconfigure.EnableAutoConfiguration=自動配置類全路徑名稱


在我們瞭解到了SpringBoot自動配置的原理之後,我們就可以自定義一個SpringBoot Starter來快速搭建我們的開發環境了

4:自定義一個列印輸入輸出日誌的Starter

4.1 首先定義一個標記需要列印出入參日誌的註解@PrintLog

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface PrintLog {
}


4.2 定義一個存放列印日誌配置的實體類

//自動註入application配置文件中已log.switch開頭的配置參數
@ConfigurationProperties("log.switch")
public class LogProperties {
    //是否啟用列印日誌功能
    private Boolean enabled = false;
    //是否列印調用者ip
    private Boolean printIp = false;
    //是否列印調用者url
    private Boolean printUrl = false
}


4.3 定義一個@PrintLog註解的切麵類

@Aspect
public class LogAspect {
    private static final Log LOGGER = LogFactory.getLog(LogAspect.class);
    private LogProperties logProperties;
    
    @Pointcut("@annotation(com.zl.annotation.PrintLog)")
    public void Log(){}

    @Around("Log()")
    public Object doAround(ProceedingJoinPoint joinPoint) throws Throwable {
        ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
        HttpServletRequest request = attributes.getRequest();
        MethodSignature signature = (MethodSignature) joinPoint.getSignature();
        Method method = signature.getMethod();
        String methodName = method.getName();
        //列印調用url
        if (Boolean.TRUE.equals(logProperties.getPrintUrl())){
            LOGGER.info("URL:" + request.getRequestURL().toString());
        }
        //列印ip
        if (Boolean.TRUE.equals(logProperties.getPrintIp())) {
            LOGGER.info("IP :" + request.getRemoteAddr());
        }
        //列印方法
        LOGGER.info("method :" + methodName);
        //列印參數
        LOGGER.info("parameter :" + Arrays.toString(joinPoint.getArgs()));
        Object result = joinPoint.proceed();
        //列印返回結果
        LOGGER.info("return :" + JSON.toJSONString(result));
        return result;
    }
}


4.4 定義一個列印日誌的自動配置類

@Configuration
@EnableConfigurationProperties({LogProperties.class})
//表示在application配置文件中必須配置log.switch.enabled = true才啟動自動配置
@ConditionalOnProperty(prefix = "log.switch", value = "enabled", havingValue = "true")
public class LogAutoConfigure {
    @Bean
    //Advice.class是aop切麵中關鍵的切麵方法類(@Before,@After等)
    //程式中有Advice.class類說明需要使用切麵功能,這時才載入自定義的切麵類
    @ConditionalOnClass(Advice.class)
    public LogAspect webLogAspect(LogProperties logProperties){
        return new LogAspect(logProperties);
    }
}


@ConditionalOnProperty表示在application配置文件中必須存在相應的配置才能使條件成立

4.5 配置自定義配置類的載入路徑

META-INF/spring.factories
org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
  com.zl.autoConfigure.LogAutoConfigure


4.6 Maven打包部署

maven install


5:開始使用自定義的Starter

5.1 在項目中引入Starter

<dependency>
   <groupId>com.zl.demo</groupId>
   <artifactId>LogStarter</artifactId>
   <version>1.0-SNAPSHOT</version>
</dependency>


5.2 在application.yml中配置參數

log:
  switch:
    enabled: true //啟用列印日誌功能
    printIp: true //列印請求ip
    printUrl: true //列印請求url


經過上面兩個步驟就,列印日誌的功能就已經開啟了,接下來就可以進行編碼測試了

5.3 定義一個Controller並標上列印日誌的註解

@RestController
@RequestMapping("/test")
public class HelloWorldController {
    @PrintLog
    @RequestMapping("/hello")
    public String helleWorld(String test){
        return "hello world!";
    }
}


5.4 啟動項目開始測試

com.zl.aspect.LogAspect   : URL:http://localhost:8080/test/hello
com.zl.aspect.LogAspect   : IP :0:0:0:0:0:0:0:1
com.zl.aspect.LogAspect   : method :helleWorld
com.zl.aspect.LogAspect   : parameter :[test]
com.zl.aspect.LogAspect   : return :"hello world!"


可以看到上面的入參和返回值都已經列印出來了,說明瞭自定義的starter已經生效了。

6:總結

SpringBoot自動配置功能帶給我們的是開箱即用,快速便捷的功能,自動配置為我們研發人員帶來的優點,我主要總結為以下兩點:

1:提高研發效率。我們可以快速構建開發環境,對於開發中使用到的開源組件和中間件,我們直接引入對應的Starter就可以直接開發了,例如Redis和Mybatis等,可以直接引入對應的_spring-boot-starter-data-redis就可以直接使用RedisTemplate來操作Redis了,這樣可以極大的提高研發的效率,無需再進行複雜的起步配置了和各種版本依賴管理了。

2:標準模塊復用。對於業務開發中的一些_標準模塊,例如常用的一些三方服務,我們可以利用Starter直接配置好,在需要使用的項目中直接引入這個starter就可以立即使用了,無需再去引入Jar包和編寫配置文件等,同樣的,對於一些標準非業務強耦合的功能_,例如監控,鑒權等,也可以定義一個Starter,需要使用鑒權和監控功能的項目就可以直接復用了,無需再次開發。

作者:京東零售 鐘磊

來源:京東雲開發者社區 自猿其說Tech 轉載請註明來源


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

-Advertisement-
Play Games
更多相關文章
  • Synchronized Synchronized關鍵字回顧 synchronized是java中的關鍵字,是一種同步鎖。它修飾的對象有以下幾種: 1.修飾一個代碼塊,被修飾的代碼塊稱為同步代碼塊,其作用的範圍是大括弧{},括起來的代碼,作用的對象是調用這個代碼塊的對象,synchronized不能 ...
  • 1. BaseHTTPRequestHandler介紹 BaseHTTPRequestHandler是Python中的一個基類,屬於http.server模塊,用於處理HTTP請求的基本功能。它提供了處理常見HTTP請求方法(如GET、POST等)的預設實現,並允許你在子類中進行定製化擴展。下麵詳細 ...
  • Dart語言和其他面向對象語言一樣,子類可以繼承父類,獲得父類的屬性和方法。那麼Dart語言,類繼承還有什麼特性呢…… ...
  • 源碼位置: /Lib/zipfile.py/ZipFile/_extract_member/zipfile.py或者直接點擊extract函數. 在使用python解壓縮zip文件時, 由於需要在解壓時重命名文件為我想要的格式, 而不巧的是, zipfile包官方源代碼沒有這個功能... 於是, 在 ...
  • 起因 在Leetcode上做題寫了兩種暴力解法,但是執行效率上不太一樣。 時間上差很遠,記憶體雖然差不多但是前者擊敗30%,後者擊敗94%。這兩種解法區別是用一條ArrayList還是兩條來存數據,所以contains雖然執行次數一樣但是檢測的長度上不一樣,而且ArrayList的擴容次數也不一樣,所 ...
  • 哈嘍大家好,我是鹹魚 我們知道,python 腳本或者說 python 程式其實是一個包含了 python 代碼的文件。要讓它們實現特定功能,我們需要知道該如何運行(run)它 通過運行 python 代碼,我們可以驗證腳本/程式是否按照我們的期望執行。這也使我們能夠對其進行測試和調試,以便找到並修 ...
  • Python 是一種面向對象的編程語言。在 Python 中,幾乎所有東西都是對象,都具有其屬性和方法。 類似於對象構造函數或用於創建對象的“藍圖”的類。 創建一個類 要創建一個類,請使用關鍵字 class: 示例,創建一個名為 MyClass 的類,其中包含一個名為 x 的屬性: class My ...
  • Github OAuth 第三方登錄示例 1、第三方登錄原理 第三方登錄的原理是藉助OAuth授權來實現,首先用戶先向客戶端提供第三方網站的數據證明自己的身份獲取授權碼,然後客戶端拿著授權碼與授權伺服器建立連接獲得一個Access Token,之後客戶端就可以通過Access Token來與資源服務 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...