厲害了!SpringBoot是如何動起來的!

来源:https://www.cnblogs.com/yunxi520/archive/2019/12/31/12127342.html
-Advertisement-
Play Games

程式入口 SpringApplication.run(BeautyApplication.class, args); 執行此方法來載入整個SpringBoot的環境。 1. 從哪兒開始? SpringApplication.java /** * Run the Spring application, ...


程式入口

SpringApplication.run(BeautyApplication.class, args);

 

執行此方法來載入整個SpringBoot的環境。

1. 從哪兒開始?

SpringApplication.java

  /**
     * Run the Spring application, creating and refreshing a new
     * {@link ApplicationContext}.
     * @param args the application arguments (usually passed from a Java main method)
     * @return a running {@link ApplicationContext}
     */
    public ConfigurableApplicationContext run(String... args) {
        //...
  }

 

調用SpringApplication.java 中的 run 方法,目的是載入Spring Application,同時返回 ApplicationContext。

2. 執行了什麼?

2.1 計時

記錄整個Spring Application的載入時間!

StopWatch stopWatch = new StopWatch();
stopWatch.start();
// ...
stopWatch.stop();
if (this.logStartupInfo) {
    new StartupInfoLogger(this.mainApplicationClass)
            .logStarted(getApplicationLog(), stopWatch);
}

 

2.2 聲明

// 聲明 ApplicationContext
ConfigurableApplicationContext context = null;
// 聲明 一個異常報告集合
Collection<SpringBootExceptionReporter> exceptionReporters = new ArrayList<>();

 

2.3 指定程式運行模式

指定 java.awt.headless,預設是true 一般是在程式開始激活headless模式,告訴程式,現在你要工作在Headless mode下,就不要指望硬體幫忙了,你得自力更生,依靠系統的計算能力模擬出這些特性來。

private void configureHeadlessProperty() {
    System.setProperty(SYSTEM_PROPERTY_JAVA_AWT_HEADLESS, System.getProperty(
            SYSTEM_PROPERTY_JAVA_AWT_HEADLESS, Boolean.toString(this.headless)));
}

 

2.4 配置監聽併發布應用啟動事件

SpringApplicationRunListener 負責載入 ApplicationListener事件。

SpringApplicationRunListeners listeners = getRunListeners(args);
// 開始
listeners.starting();
// 處理所有 property sources 配置和 profiles 配置,準備環境,分為標準 Servlet 環境和標準環境
ConfigurableEnvironment environment = prepareEnvironment(listeners,applicationArguments);
// 準備應用上下文
prepareContext(context, environment, listeners, applicationArguments,printedBanner);
// 完成
listeners.started(context);
// 異常
handleRunFailure(context, ex, exceptionReporters, listeners);
// 執行
listeners.running(context);

 

getRunListeners 中根據 type = SpringApplicationRunListener.class 去拿到了所有的 Listener 並根據優先順序排序。

對應的就是 META-INF/spring.factories 文件中的 org.springframework.boot.SpringApplicationRunListener=org.springframework.boot.context.event.EventPublishingRunListener

private <T> Collection<T> getSpringFactoriesInstances(Class<T> type,
            Class<?>[] parameterTypes, Object... args) {
        ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
        // Use names and ensure unique to protect against duplicates
        Set<String> names = new LinkedHashSet<>(
                SpringFactoriesLoader.loadFactoryNames(type, classLoader));
        List<T> instances = createSpringFactoriesInstances(type, parameterTypes,
                classLoader, args, names);
        AnnotationAwareOrderComparator.sort(instances);
        return instances;
    }

 

在 ApplicationListener 中 , 可以針對任何一個階段插入處理代碼。

public interface SpringApplicationRunListener {

    /**
     * Called immediately when the run method has first started. Can be used for very
     * early initialization.
     */
    void starting();

    /**
     * Called once the environment has been prepared, but before the
     * {@link ApplicationContext} has been created.
     * @param environment the environment
     */
    void environmentPrepared(ConfigurableEnvironment environment);

    /**
     * Called once the {@link ApplicationContext} has been created and prepared, but
     * before sources have been loaded.
     * @param context the application context
     */
    void contextPrepared(ConfigurableApplicationContext context);

    /**
     * Called once the application context has been loaded but before it has been
     * refreshed.
     * @param context the application context
     */
    void contextLoaded(ConfigurableApplicationContext context);

    /**
     * The context has been refreshed and the application has started but
     * {@link CommandLineRunner CommandLineRunners} and {@link ApplicationRunner
     * ApplicationRunners} have not been called.
     * @param context the application context.
     * @since 2.0.0
     */
    void started(ConfigurableApplicationContext context);

    /**
     * Called immediately before the run method finishes, when the application context has
     * been refreshed and all {@link CommandLineRunner CommandLineRunners} and
     * {@link ApplicationRunner ApplicationRunners} have been called.
     * @param context the application context.
     * @since 2.0.0
     */
    void running(ConfigurableApplicationContext context);

    /**
     * Called when a failure occurs when running the application.
     * @param context the application context or {@code null} if a failure occurred before
     * the context was created
     * @param exception the failure
     * @since 2.0.0
     */
    void failed(ConfigurableApplicationContext context, Throwable exception);

}

 

3. 每個階段執行的內容

3.1 listeners.starting();

在載入Spring Application之前執行,所有資源和環境未被載入。

3.2 prepareEnvironment(listeners, applicationArguments);

創建 ConfigurableEnvironment;將配置的環境綁定到Spring Application中;

    private ConfigurableEnvironment prepareEnvironment(
            SpringApplicationRunListeners listeners,
            ApplicationArguments applicationArguments) {
        // Create and configure the environment
        ConfigurableEnvironment environment = getOrCreateEnvironment();
        configureEnvironment(environment, applicationArguments.getSourceArgs());
        listeners.environmentPrepared(environment);
        bindToSpringApplication(environment);
        if (this.webApplicationType == WebApplicationType.NONE) {
            environment = new EnvironmentConverter(getClassLoader())
                    .convertToStandardEnvironmentIfNecessary(environment);
        }
        ConfigurationPropertySources.attach(environment);
        return environment;
    }

 

3.3 prepareContext

配置忽略的Bean;

 private void configureIgnoreBeanInfo(ConfigurableEnvironment environment) {
        if (System.getProperty(
                CachedIntrospectionResults.IGNORE_BEANINFO_PROPERTY_NAME) == null) {
            Boolean ignore = environment.getProperty("spring.beaninfo.ignore",
                    Boolean.class, Boolean.TRUE);
            System.setProperty(CachedIntrospectionResults.IGNORE_BEANINFO_PROPERTY_NAME,
                    ignore.toString());
        }
    }

 

列印日誌-載入的資源

Banner printedBanner = printBanner(environment);

 

根據不同的WebApplicationType創建Context

context = createApplicationContext();

 

3.4 refreshContext

支持定製刷新

 /**
     * Register a shutdown hook with the JVM runtime, closing this context
     * on JVM shutdown unless it has already been closed at that time.
     * <p>This method can be called multiple times. Only one shutdown hook
     * (at max) will be registered for each context instance.
     * @see java.lang.Runtime#addShutdownHook
     * @see #close()
     */
    void registerShutdownHook();

 

3.5 afterRefresh

刷新後的實現方法暫未實現

 /**
     * Called after the context has been refreshed.
     * @param context the application context
     * @param args the application arguments
     */
    protected void afterRefresh(ConfigurableApplicationContext context,
            ApplicationArguments args) {
    }

 

3.6 listeners.started(context);

到此為止, Spring Application的環境和資源都載入完畢了;發佈應用上下文啟動完成事件;執行所有 Runner 運行器 - 執行所有 ApplicationRunner 和 CommandLineRunner 這兩種運行器。

// 啟動
callRunners(context, applicationArguments);

 

3.7 listeners.running(context);

觸發所有 SpringApplicationRunListener 監聽器的 running 事件方法

 

(完)

本人免費整理了Java高級資料,涵蓋了Java、Redis、MongoDB、MySQL、Zookeeper、Spring Cloud、Dubbo高併發分散式等教程,一共30G,需要自己領取。
傳送門:https://mp.weixin.qq.com/s/osB-BOl6W-ZLTSttTkqMPQ


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

-Advertisement-
Play Games
更多相關文章
  • JS內置對象【字元串】 // charAt() 返回字元 // charCodeAt() 返回字元的unicode編碼 var str="hello world"; console.log(str.charCodeAt(4)); // indexOf() 返回出現的位置 // 沒找到則返回-1 va ...
  • 12306 結合日曆時間,推出了“按小時為基準的定時、批量、放票的策略,先到者得”的系統邏輯。 這就導致,所有購票人,都要在搶在某個時間點,比如 10:00:00 , 十幾秒內,或者最好在零點幾秒內,完成購票操作。 ... “電腦隨機抽簽模式”的好處在於: 購票人可以不用急急忙忙地購票,... ...
  • 昨天我們聊了 架構是什麼 ?今天我們來看看架構的前世今生。客戶端-伺服器(C/S)、瀏覽器-伺服器(B/S)、面向服務架構(SOA)、微服務(Microservice)、無伺服器(Serverless)、函數計算(FaaS)等,我們都聽說過不少架構相關的名詞,但它們之間究竟是什麼關係,哪種架構更好?... ...
  • 架構師,老兵哥剛參加工作那些年業界還沒有這個職位,那時候跟技術相關的崗位就是開發工程師、測試工程師和系統工程師,後來隨著軟體規模不斷增長而產生的,尤其是在互聯網浪潮下用戶數和訪問量都是海量化的。在各種機緣巧合下,老兵哥結合個人喜好選擇了走架構師路徑,從懵懵懂懂邊做邊學,到現在總算摸出了些門道,回顧這... ...
  • 1.泛型類 普通的類 這樣的代碼是完全可以執行了,那為什麼還需要泛型類? 1.安全性 上面的代碼編譯是完全可以通過的,但是執行的時候就會出現ClassCastException異常 2.可讀性好,省去了反覆的強制類型轉換。 對於泛型類,java編譯器會將泛型代碼轉換成普通的非泛型代碼, 所以對於虛擬 ...
  • 一、Spring Data JPA 1、簡介 (1)官網地址: https://spring.io/projects/spring-data-jpa參考文檔: https://docs.spring.io/spring-data/jpa/docs/2.2.3.RELEASE/reference/ht ...
  • 本篇文章主要介紹PHP+swoole實現聊天群發功能,感興趣的朋友參考下,希望對大家有所幫助。 php代碼: $serv = new swoole_websocket_server("127.0.0.1",3999); //服務的基本設置 $serv->set(array( 'worker_num' ...
  • 這篇文章主要介紹了關於PHP實現微信網頁登陸授權開發,有著一定的參考價值,現在分享給大家,有需要的朋友可以參考一下 更多PHP相關知識請關註我的專欄PHP​zhuanlan.zhihu.com 微信開放平臺和公眾平臺的區別 1.公眾平臺面向的時普通的用戶,比如自媒體和媒體,企業官方微信公眾賬號運營人 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...