SpringBoot啟動流程分析1:SpringApplication類初始化過程

目錄

一、SpringApplication初始化過程

1.1狞洋、SpringBoot項目的main函數(shù)
1.2弯淘、 SpringApplication() 構(gòu)造方法

  • 1.2.1吉懊、deduceWebApplicationType();
  • 1.2.2、 setInitializers((Collection) getSpringFactoriesInstances(ApplicationContextInitializer.class));
  • 1.2.3借嗽、 setListeners((Collection) getSpringFactoriesInstances(ApplicationListener.class));

二怕午、總結(jié)

一、SpringApplication初始化過程

1.1淹魄、SpringBoot項目的main函數(shù)

常規(guī)的這個主類如下圖所示,我們一般會這樣去寫甲锡。

@SpringBootApplication
public class SpringbootApplication {

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

}

在這個類中需要關(guān)注的是

  • @SpringBootApplication
  • SpringApplication.run()

關(guān)于 @SpringBootApplication 注解,在后面分析SpringBoot自動裝配的章節(jié)會展開去分析缤沦。

本章節(jié)中我們需要關(guān)注的就是 SpringApplication.run() 方法。

查看run()方法的實現(xiàn)包蓝,如下面代碼所示企量,我們發(fā)現(xiàn)其實其首先是創(chuàng)建了 SpringApplication 的實例,然后調(diào)用了 SpringApplication 的run()方法届巩,那本章我們關(guān)注的就是 SpringApplication 創(chuàng)建實例的過程。

/**
     * Static helper that can be used to run a {@link SpringApplication} from the
     * specified sources using default settings and user supplied arguments.
     *
     * @param primarySources the primary sources to load
     * @param args           the application arguments (usually passed from a Java main method)
     * @return the running {@link ApplicationContext}
     */
    public static ConfigurableApplicationContext run(Class<?> primarySource, String... args) {
        return run(new Class[]{primarySource}, args);
    }

1.2腕唧、 SpringApplication() 構(gòu)造方法

    public SpringApplication(ResourceLoader resourceLoader, Class<?>... primarySources) {
        this.sources = new LinkedHashSet();
        this.bannerMode = Mode.CONSOLE;
        this.logStartupInfo = true;
        this.addCommandLineProperties = true;
        this.addConversionService = true;
        this.headless = true;
        this.registerShutdownHook = true;
        this.additionalProfiles = new HashSet();
        this.isCustomEnvironment = false;
        this.lazyInitialization = false;
        this.resourceLoader = resourceLoader;
        Assert.notNull(primarySources, "PrimarySources must not be null");
        this.primarySources = new LinkedHashSet(Arrays.asList(primarySources));
        this.webApplicationType = WebApplicationType.deduceFromClasspath();
        this.setInitializers(this.getSpringFactoriesInstances(ApplicationContextInitializer.class));
        this.setListeners(this.getSpringFactoriesInstances(ApplicationListener.class));
        this.mainApplicationClass = this.deduceMainApplicationClass();
    }

1.2.1枣接、deduceWebApplicationType();

該方法推斷應(yīng)用的類型缺谴。 SERVLET REACTIVE NONE

    private static final String[] SERVLET_INDICATOR_CLASSES = new String[]{"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";
    private static final String SERVLET_APPLICATION_CONTEXT_CLASS = "org.springframework.web.context.WebApplicationContext";
    private static final String REACTIVE_APPLICATION_CONTEXT_CLASS = "org.springframework.boot.web.reactive.context.ReactiveWebApplicationContext";

    private WebApplicationType() {
    }

    static WebApplicationType deduceFromClasspath() {
        if (ClassUtils.isPresent("org.springframework.web.reactive.DispatcherHandler", (ClassLoader)null) && !ClassUtils.isPresent("org.springframework.web.servlet.DispatcherServlet", (ClassLoader)null) && !ClassUtils.isPresent("org.glassfish.jersey.servlet.ServletContainer", (ClassLoader)null)) {
            return REACTIVE;
        } else {
            String[] var0 = SERVLET_INDICATOR_CLASSES;
            int var1 = var0.length;

            for(int var2 = 0; var2 < var1; ++var2) {
                String className = var0[var2];
                if (!ClassUtils.isPresent(className, (ClassLoader)null)) {
                    return NONE;
                }
            }

            return SERVLET;
        }
    }

返回類型是WebApplicationType的枚舉類型瓣赂, WebApplicationType 有三個枚舉,三個枚舉的解釋如其中注釋

具體的判斷邏輯如下:

  • WebApplicationType.REACTIVE classpath下存在org.springframework.web.reactive.DispatcherHandler

  • WebApplicationType.SERVLET classpath下存在javax.servlet.Servlet或者org.springframework.web.context.ConfigurableWebApplicationContext

  • WebApplicationType.NONE 不滿足以上條件妓肢。

1.2.2苫纤、 setInitializers((Collection) getSpringFactoriesInstances(ApplicationContextInitializer.class));

初始化classpath下 META-INF/spring.factories中已配置的ApplicationContextInitializer。

/**
  * 通過指定的classloader 從META-INF/spring.factories獲取指定的Spring的工廠實例
  * @param type
  * @param parameterTypes
  * @param args
  * @param <T>
  * @return
  */
    private <T> Collection<T> getSpringFactoriesInstances(Class<T> type, Class<?>[] parameterTypes, Object... args) {
        ClassLoader classLoader = this.getClassLoader();
        // Use names and ensure unique to protect against duplicates
        //通過指定的classLoader從 META-INF/spring.factories 的資源文件中喊废,
        //讀取 key 為 type.getName() 的 value
        Set<String> names = new LinkedHashSet(SpringFactoriesLoader.loadFactoryNames(type, classLoader));
        List<T> instances = this.createSpringFactoriesInstances(type, parameterTypes, classLoader, args, names);
        AnnotationAwareOrderComparator.sort(instances);
        return instances;
    }

看看 getSpringFactoriesInstances 都干了什么栗弟,看源碼,有一個方法很重要 loadFactoryNames() 這個方法很重要乍赫,這個方法是spring-core中提供的從META-INF/spring.factories中獲取指定的類(key)的同一入口方法陆蟆。

在這里惋增,獲取的是key為 org.springframework.context.ApplicationContextInitializer 的類。

debug看看都獲取到了哪些

image

上面說了林束,是從classpath下 META-INF/spring.factories中獲取稽亏,我們驗證一下:

image
image

發(fā)現(xiàn)在上圖所示的兩個工程中找到了debug中看到的6條結(jié)果措左。 ApplicationContextInitializer 是Spring框架的類, 這個類的主要目的就是在 ConfigurableApplicationContext 調(diào)用refresh()方法之前,回調(diào)這個類的initialize方法胸嘁。通過 ConfigurableApplicationContext 的實例獲取容器的環(huán)境Environment凉逛,從而實現(xiàn)對配置文件的修改完善等工作。

關(guān)于怎么實現(xiàn)自定義的 ApplicationContextInitializer 請看我的另一篇專門介紹該類的博客状飞。

1.2.3、 setListeners((Collection) getSpringFactoriesInstances(ApplicationListener.class));

初始化classpath下 META-INF/spring.factories中已配置的 ApplicationListener酵使。

ApplicationListener 的加載過程和上面的 ApplicationContextInitializer 類的加載過程是一樣的焙糟。不多說了,至于 ApplicationListener 是spring的事件監(jiān)聽器穿撮,典型的觀察者模式,通過 ApplicationEvent 類和 ApplicationListener 接口攻礼,可以實現(xiàn)對spring容器全生命周期的監(jiān)聽栗柒,當然也可以自定義監(jiān)聽事件。為了梳理springboot的啟動流程在這里先不說這個了。后面有時間的話再介紹负蠕。

關(guān)于ApplicationContextInitializer的詳細介紹請看<SpringBoot之ApplicationContextInitializer的理解和使用>

二倦畅、總結(jié)

關(guān)于 SpringApplication 類的構(gòu)造過程绣的,到這里我們就梳理完了÷沤縱觀 SpringApplication 類的實例化過程,我們可以看到罢洲,合理的利用該類文黎,我們能在spring容器創(chuàng)建之前做一些預(yù)備工作,和定制化的需求桩蓉。

比如,自定義SpringBoot的Banner院究,比如自定義事件監(jiān)聽器本涕,再比如在容器refresh之前通過自定義 ApplicationContextInitializer 修改配置一些配置或者獲取指定的bean都是可以的。菩颖。。

下一節(jié)開始分析SpringBoot容器的構(gòu)建過程氛濒,也就是那個大家多少都看過的run();方法鹅髓。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市骗奖,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌执桌,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,194評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件伴逸,死亡現(xiàn)場離奇詭異膘壶,居然都是意外死亡,警方通過查閱死者的電腦和手機顷锰,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,058評論 2 385
  • 文/潘曉璐 我一進店門亡问,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人束世,你說我怎么就攤上這事慎框。” “怎么了笨枯?”我有些...
    開封第一講書人閱讀 156,780評論 0 346
  • 文/不壞的土叔 我叫張陵馅精,是天一觀的道長。 經(jīng)常有香客問我洲敢,道長,這世上最難降的妖魔是什么睦优? 我笑而不...
    開封第一講書人閱讀 56,388評論 1 283
  • 正文 為了忘掉前任壮不,我火速辦了婚禮,結(jié)果婚禮上隐孽,老公的妹妹穿的比我還像新娘癌椿。我一直安慰自己菱阵,他們只是感情好,可當我...
    茶點故事閱讀 65,430評論 5 384
  • 文/花漫 我一把揭開白布都办。 她就那樣靜靜地躺著虑稼,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上跟压,一...
    開封第一講書人閱讀 49,764評論 1 290
  • 那天震蒋,我揣著相機與錄音,去河邊找鬼查剖。 笑死,一個胖子當著我的面吹牛笋庄,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播菌仁,決...
    沈念sama閱讀 38,907評論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼静暂,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了摹迷?” 一聲冷哼從身側(cè)響起郊供,我...
    開封第一講書人閱讀 37,679評論 0 266
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎异赫,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體塔拳,經(jīng)...
    沈念sama閱讀 44,122評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,459評論 2 325
  • 正文 我和宋清朗相戀三年量九,在試婚紗的時候發(fā)現(xiàn)自己被綠了颂碧。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,605評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡肌似,死狀恐怖诉瓦,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情固额,我是刑警寧澤,帶...
    沈念sama閱讀 34,270評論 4 329
  • 正文 年R本政府宣布斗躏,位于F島的核電站昔脯,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏迈套。R本人自食惡果不足惜碱鳞,卻給世界環(huán)境...
    茶點故事閱讀 39,867評論 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望窿给。 院中可真熱鬧,春花似錦禁荒、人聲如沸角撞。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,734評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至姐军,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間奕锌,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,961評論 1 265
  • 我被黑心中介騙來泰國打工饼丘, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留辽话,地道東北人。 一個月前我還...
    沈念sama閱讀 46,297評論 2 360
  • 正文 我出身青樓,卻偏偏與公主長得像忽肛,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子础废,可洞房花燭夜當晚...
    茶點故事閱讀 43,472評論 2 348