【java】(一)SpringBoot 源碼解析——SpringApplication 初始化

来源:https://www.cnblogs.com/anlizhaomi/archive/2022/12/12/16956073.html
-Advertisement-
Play Games

1.前言 深入學習springboot筆記系列,可能會有錯誤還請指正,互相勉勵,互相學習。 SpringBoot 項目啟動只需啟動 主類的 main 函數即可啟動java服務,相比於以往的部署java服務簡化方便了很多,接下我們從主函數入手一步一步剖析源碼是如何通過main函數啟動服務的。 2.Sp ...


1.前言

深入學習springboot筆記系列,可能會有錯誤還請指正,互相勉勵,互相學習。

SpringBoot 項目啟動只需啟動 主類的 main 函數即可啟動java服務,相比於以往的部署java服務簡化方便了很多,接下我們從主函數入手一步一步剖析源碼是如何通過main函數啟動服務的。

2.SpringBoot 項目程式入口

主函數通過一個靜態 run 方法完成整個服務的構建。

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

接下來看看靜態的 run 方法的內部實現。

2.1.run 方法構造

public static ConfigurableApplicationContext run(Class<?> primarySource, String... args) {
return run(new Class<?>[] { primarySource }, args);
}
public static ConfigurableApplicationContext run(Class<?>[] primarySources, String[] args) {
return new SpringApplication(primarySources).run(args);
}

以上通過 new SpringApplication(primarySources)  執行了初始化的一些相關操作。

3.SpringApplication 初始化

public SpringApplication(Class<?>... primarySources) {
  this(null, primarySources);
}

public SpringApplication(ResourceLoader resourceLoader, Class<?>... primarySources) {
  this.resourceLoader = resourceLoader;
  Assert.notNull(primarySources, "PrimarySources must not be null");
  this.primarySources = new LinkedHashSet<>(Arrays.asList(primarySources));
  // 推斷服務類型
  this.webApplicationType = WebApplicationType.deduceFromClasspath();
  // 初始化 META-INF/spring.factories 中 所有的 BootstrapRegistryInitializer 類
  this.bootstrapRegistryInitializers = new ArrayList<>(getSpringFactoriesInstances(BootstrapRegistryInitializer.class));
  // 初始化 META-INF/spring.factories 中 所有的 ApplicationContextInitializer 類
  setInitializers((Collection) getSpringFactoriesInstances(ApplicationContextInitializer.class));
  // 初始化 META-INF/spring.factories 中 所有的 ApplicationListener 類
  setListeners((Collection) getSpringFactoriesInstances(ApplicationListener.class));
  // 推斷主類
  this.mainApplicationClass = deduceMainApplicationClass();
}

SpringApplication 初始化主要初始化了 resourceLoader、primarySources、webApplicationType 、bootstrapRegistryInitializers、initializers、listeners、mainApplicationClass 這幾個對象。

其中resourceLoader 預設為null, primarySources 則為主類的有序去重集合。

3.1.webApplicationType 推斷當前項目啟動類型

private static final String[] SERVLET_INDICATOR_CLASSES = { "javax.servlet.Servlet",
            "org.springframework.web.context.ConfigurableWebApplicationContext" };

private static final String WEBMVC_INDICATOR_CLASS = "org.springframework.web.servlet.DispatcherServlet";

private static final String WEBFLUX_INDICATOR_CLASS = "org.springframework.web.reactive.DispatcherHandler";

private static final String JERSEY_INDICATOR_CLASS = "org.glassfish.jersey.servlet.ServletContainer";


static WebApplicationType deduceFromClasspath() {
//webflux 響應式
if (ClassUtils.isPresent(WEBFLUX_INDICATOR_CLASS, null) && !ClassUtils.isPresent(WEBMVC_INDICATOR_CLASS, null) && !ClassUtils.isPresent(JERSEY_INDICATOR_CLASS, null)) { return WebApplicationType.REACTIVE; }
// 是否web 項目
for (String className : SERVLET_INDICATOR_CLASSES) { if (!ClassUtils.isPresent(className, null)) { return WebApplicationType.NONE; } } return WebApplicationType.SERVLET; }

此處使用 ClassUtils.isPresent(String className, @Nullable ClassLoader classLoader) 方法為內部調用類載入器載入相應的類,如果未找到則拋出ClassNotFoundException異常。類載入 WEBMVC_INDICATOR_CLASS、WEBFLUX_INDICATOR_CLASS 、JERSEY_INDICATOR_CLASS 幾種不同的 Servlet,如果未載入到則為false,從而推斷項目啟動類型。此處我們是web項目,因此最後的返回值是WebApplicationType.SERVLET。

3.2.bootstrapRegistryInitializers 初始化


  private <T> Collection<T> getSpringFactoriesInstances(Class<T> type) {

return getSpringFactoriesInstances(type, new Class<?>[] {});
}

private <T> Collection<T> getSpringFactoriesInstances(Class<T> type, Class<?>[] parameterTypes, Object... args) {
//獲取預設類載入器AppClassLoader ClassLoader classLoader
= getClassLoader(); //使用預設類載入器載入META-INFO/spring.factories 中配置的類 Set<String> names = new LinkedHashSet<>(SpringFactoriesLoader.loadFactoryNames(type, classLoader));
//使用反射將上一步 類載入完成的 names中的類實例化 List
<T> instances = createSpringFactoriesInstances(type, parameterTypes, classLoader, args, names);
// 將實例化的類排序 排序規則:繼承了 PriorityOrdered 介面的 優先順序最高, 其次實現Ordered 介面 或者添加@Order 註解 通過比較order值的大小來排序,值越小優先順序越高。 AnnotationAwareOrderComparator.sort(instances);
return instances; } @SuppressWarnings("unchecked") private <T> List<T> createSpringFactoriesInstances(Class<T> type, Class<?>[] parameterTypes, ClassLoader classLoader, Object[] args, Set<String> names) { List<T> instances = new ArrayList<>(names.size()); for (String name : names) { try { Class<?> instanceClass = ClassUtils.forName(name, classLoader); Assert.isAssignable(type, instanceClass); Constructor<?> constructor = instanceClass.getDeclaredConstructor(parameterTypes); T instance = (T) BeanUtils.instantiateClass(constructor, args); instances.add(instance); } catch (Throwable ex) { throw new IllegalArgumentException("Cannot instantiate " + type + " : " + name, ex); } } return instances; } public static List<String> loadFactoryNames(Class<?> factoryType, @Nullable ClassLoader classLoader) { ClassLoader classLoaderToUse = classLoader; if (classLoaderToUse == null) { classLoaderToUse = SpringFactoriesLoader.class.getClassLoader(); } String factoryTypeName = factoryType.getName(); return loadSpringFactories(classLoaderToUse).getOrDefault(factoryTypeName, Collections.emptyList()); }
private static Map<String, List<String>> loadSpringFactories(ClassLoader classLoader) {
// 緩存cache 中查找 Map
<String, List<String>> result = cache.get(classLoader); if (result != null) { return result; } result = new HashMap<>(); try {
// 此處使用類載入器讀取所有依賴的包中 MEAT-INFO/spring.factories 中配置的類,並添加在緩存cache中 Enumeration
<URL> urls = classLoader.getResources(FACTORIES_RESOURCE_LOCATION); while (urls.hasMoreElements()) { URL url = urls.nextElement(); UrlResource resource = new UrlResource(url); Properties properties = PropertiesLoaderUtils.loadProperties(resource); for (Map.Entry<?, ?> entry : properties.entrySet()) { String factoryTypeName = ((String) entry.getKey()).trim(); String[] factoryImplementationNames = StringUtils.commaDelimitedListToStringArray((String) entry.getValue()); for (String factoryImplementationName : factoryImplementationNames) { result.computeIfAbsent(factoryTypeName, key -> new ArrayList<>()) .add(factoryImplementationName.trim()); } } } // Replace all lists with unmodifiable lists containing unique elements result.replaceAll((factoryType, implementations) -> implementations.stream().distinct() .collect(Collectors.collectingAndThen(Collectors.toList(), Collections::unmodifiableList))); cache.put(classLoader, result); } catch (IOException ex) { throw new IllegalArgumentException("Unable to load factories from location [" + FACTORIES_RESOURCE_LOCATION + "]", ex); } return result; }

以上方法通過預設的類載入AppClassLoader 載入依賴包中 META-INFO/spring.factories 路徑中所有配置的類並緩存在cache中,然後由class 類型找出緩存cache 中需要相應載入的類並通過反射將類實例化併排序返回。

3.3.initializers 、listeners 初始化

同 bootstrapRegistryInitializers 載入流程一致,通過cache 緩存提高載入速度。

3.4.mainApplicationClass 初始化

private Class<?> deduceMainApplicationClass() {
    try {
        StackTraceElement[] stackTrace = new RuntimeException().getStackTrace();
        for (StackTraceElement stackTraceElement : stackTrace) {
            if ("main".equals(stackTraceElement.getMethodName())) {
                return Class.forName(stackTraceElement.getClassName());
            }
        }
    }
    catch (ClassNotFoundException ex) {
        // Swallow and continue
    }
    return null;
}

通過構造一個運行異常獲取堆棧信息,從main方法的堆棧信息中獲取主類的信息。因為SpringApplication的構造器primarySources 是數組類型,因此無法直接通過primarySource 來判斷main方法屬於哪個class。

至此 SpringApplication類初始化載入完成。

總結

1.通過類載入器載入依賴的servlet判斷項目類型;

2.通過類載入器載入指定路徑文件 MEAT-INFO/spring.factories 讀取指定配置文件並使用cache 緩存提高載入速度;

3.通過java反射的方式將指定的 BootstrapRegistryInitializer.class、ApplicationContextInitializer.classApplicationListener.class 類的實現類實例化;

4.通過運行異常的堆棧信息推斷main方法所在的類為主類。

 

後續

下一章 將繼續 講解 SpringBoot 後續啟動流程,謝謝觀看。


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

-Advertisement-
Play Games
更多相關文章
  • 多線程程式 競態條件:多線程程式執行的結果是一致的,不會隨著CPU對線程不同的調用順序而產生不同的運行結果. 解決?:互斥鎖 mutex 經典的賣票問題,三個線程賣100張票 代碼1 #include <iostream> #include <thread> #include <list> #inc ...
  • C++語言層面多線程=>好處:跨平臺 windows/linux thread/mutex/condition_variable lock_gurad/unique_lock atomic/原子類型,基於CAS操作的原子類型 線程安全的 睡眠sleep_for C++ thread => windo ...
  • JZ45 把數組排成最小的數 描述 輸入一個非負整數數組numbers,把數組裡所有數字拼接起來排成一個數,列印能拼接出的所有數字中最小的一個。 例如輸入數組[3,32,321],則列印出這三個數字能排成的最小數字為321323。 1.輸出結果可能非常大,所以你需要返回一個字元串而不是整數 2.拼接 ...
  • 1. String 字元串是 Redis 最基本的數據類型,不僅所有 key 都是字元串類型,其它幾種數據類型構成的元素也是字元串。註意字元串的長度不能超過 512M。 1.1 編碼方式(encoding) 字元串對象的編碼可以是 int ,raw 或者 embstr 。 int 編碼:保存的是可以 ...
  • 應用背景: 隨著科學技術的發展,崗位數量越來越多,特別是每逢畢業季找工作的人數也很多,如果人們找工作或者企業招人靠純手工的話,費時費力,僅僅是篩選簡歷和費勁,並且員工找工作投簡歷可能得需要剋服時間和空間上的困難。所以為了方便員工找工作和企業招人,節約時間,特此開發員工招聘系統。(個人課設) 用例圖( ...
  • 原文:Jgit的使用筆記 - Stars-One的雜貨小窩 之前整的一個系統,涉及到git代碼的推送,是通過cmd命令去推送的,然後最近在產品驗收的時候,測試部門隨意填了個git倉庫,然後導致倉庫代碼被覆蓋了,還好本地留有備份,沒出現啥大問題 然後就計劃於是就改為使用Jgit庫來實現推送代碼的功能, ...
  • 哈嘍兄弟們,我們在學習Python的過程中,有這麼一款工具,可以輕鬆糾正我們寫錯的命令,簡直太好用了~ The Fuck 是一款功能強大的、Python編寫的應用程式,可用於糾正控制台命令中的錯誤,非常強大。此外,用戶還可通過寫Python代碼的方式自定義修複規則。 修複效果如下動圖所示: 更多示例 ...
  • 前言 今天給大家介紹的是Python爬取手機商品信息數據,在這裡給需要的小伙伴們代碼,並且給出一點小心得。 首先是爬取之前應該儘可能偽裝成瀏覽器而不被識別出來是爬蟲,基本的是加請求頭,但是這樣的純文本數據爬取的人會很多,所以我們需要考慮更換代理IP和隨機更換請求頭的方式來對手機信息數據進行爬取。 在 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...