Springboot詳解底層啟動(dòng)過(guò)程
SpringApplication構(gòu)造分析
1、記錄 BeanDefinition 源
spring容器剛開(kāi)始是空的,要去各個(gè)源找到beanDefinition,這些源可能是配置類,可能是xml文件。在構(gòu)造方法里會(huì)獲取一個(gè)主源,也就是引導(dǎo)類,根據(jù)引導(dǎo)類去獲取beanDefinition。
2、推斷應(yīng)用類型
根據(jù)jar包去判斷是什么引用類型
3、記錄 ApplicationContext 初始化器
對(duì)ApplicationContext做擴(kuò)展
4、記錄監(jiān)聽(tīng)器
監(jiān)聽(tīng)重要事件
5、推斷主啟動(dòng)類
記錄運(yùn)行的主類。
SpringApplication run分析
1、得到 SpringApplicationRunListeners,名字取得不好,實(shí)際是事件發(fā)布器
發(fā)布 application starting 事件,在程序啟動(dòng)的重要節(jié)點(diǎn)發(fā)布事件
public static void main(String[] args) throws Exception{
// 添加 app 監(jiān)聽(tīng)器
SpringApplication app = new SpringApplication();
app.addListeners(e -> System.out.println(e.getClass()));
// 獲取事件發(fā)送器實(shí)現(xiàn)類名
List<String> names = SpringFactoriesLoader.loadFactoryNames(SpringApplicationRunListener.class, A39_2.class.getClassLoader());
for (String name : names) {
System.out.println(name);
Class<?> clazz = Class.forName(name);
Constructor<?> constructor = clazz.getConstructor(SpringApplication.class, String[].class);
SpringApplicationRunListener publisher = (SpringApplicationRunListener) constructor.newInstance(app, args);
// 發(fā)布事件
DefaultBootstrapContext bootstrapContext = new DefaultBootstrapContext();
publisher.starting(bootstrapContext); // spring boot 開(kāi)始啟動(dòng)
publisher.environmentPrepared(bootstrapContext, new StandardEnvironment()); // 環(huán)境信息準(zhǔn)備完畢
GenericApplicationContext context = new GenericApplicationContext();
publisher.contextPrepared(context); // 在 spring 容器創(chuàng)建,并調(diào)用初始化器之后,發(fā)送此事件
publisher.contextLoaded(context); // 所有 bean definition 加載完畢
context.refresh();
publisher.started(context); // spring 容器初始化完成(refresh 方法調(diào)用完畢)
publisher.running(context); // spring boot 啟動(dòng)完畢
publisher.failed(context, new Exception("出錯(cuò)了")); // spring boot 啟動(dòng)出錯(cuò)
}2、封裝啟動(dòng) args
3、準(zhǔn)備 Environment 添加命令行參數(shù)(*)
public static void main(String[] args) throws IOException {
ApplicationEnvironment env = new ApplicationEnvironment(); // 系統(tǒng)環(huán)境變量, properties, yaml
env.getPropertySources().addLast(new ResourcePropertySource(new ClassPathResource("step3.properties")));
env.getPropertySources().addFirst(new SimpleCommandLinePropertySource(args));
for (PropertySource<?> ps : env.getPropertySources()) {
System.out.println(ps);
}
// System.out.println(env.getProperty("JAVA_HOME"));
System.out.println(env.getProperty("server.port"));
}4、ConfigurationPropertySources 處理(*)
發(fā)布 application environment 已準(zhǔn)備事件
public static void main(String[] args) throws IOException, NoSuchFieldException {
ApplicationEnvironment env = new ApplicationEnvironment();
env.getPropertySources().addLast(
new ResourcePropertySource("step4", new ClassPathResource("step4.properties"))
);
ConfigurationPropertySources.attach(env);
for (PropertySource<?> ps : env.getPropertySources()) {
System.out.println(ps);
}
System.out.println(env.getProperty("user.first-name"));
System.out.println(env.getProperty("user.middle-name"));
System.out.println(env.getProperty("user.last-name"));
}
}5、通過(guò) EnvironmentPostProcessorApplicationListener 進(jìn)行 env 后處理(*)
application.properties,由 StandardConfigDataLocationResolver 解析
spring.application.json
public class Step5 {
public static void main(String[] args) {
SpringApplication app = new SpringApplication();
app.addListeners(new EnvironmentPostProcessorApplicationListener());
/*List<String> names = SpringFactoriesLoader.loadFactoryNames(EnvironmentPostProcessor.class, Step5.class.getClassLoader());
for (String name : names) {
System.out.println(name);
}*/
EventPublishingRunListener publisher = new EventPublishingRunListener(app, args);
ApplicationEnvironment env = new ApplicationEnvironment();
System.out.println(">>>>>>>>>>>>>>>>>>>>>>>>> 增強(qiáng)前");
for (PropertySource<?> ps : env.getPropertySources()) {
System.out.println(ps);
}
publisher.environmentPrepared(new DefaultBootstrapContext(), env);
System.out.println(">>>>>>>>>>>>>>>>>>>>>>>>> 增強(qiáng)后");
for (PropertySource<?> ps : env.getPropertySources()) {
System.out.println(ps);
}
}
private static void test1() {
SpringApplication app = new SpringApplication();
ApplicationEnvironment env = new ApplicationEnvironment();
System.out.println(">>>>>>>>>>>>>>>>>>>>>>>>> 增強(qiáng)前");
for (PropertySource<?> ps : env.getPropertySources()) {
System.out.println(ps);
}
ConfigDataEnvironmentPostProcessor postProcessor1 = new ConfigDataEnvironmentPostProcessor(new DeferredLogs(), new DefaultBootstrapContext());
postProcessor1.postProcessEnvironment(env, app);
System.out.println(">>>>>>>>>>>>>>>>>>>>>>>>> 增強(qiáng)后");
for (PropertySource<?> ps : env.getPropertySources()) {
System.out.println(ps);
}
RandomValuePropertySourceEnvironmentPostProcessor postProcessor2 = new RandomValuePropertySourceEnvironmentPostProcessor(new DeferredLog());
postProcessor2.postProcessEnvironment(env, app);
System.out.println(">>>>>>>>>>>>>>>>>>>>>>>>> 增強(qiáng)后");
for (PropertySource<?> ps : env.getPropertySources()) {
System.out.println(ps);
}
System.out.println(env.getProperty("server.port"));
System.out.println(env.getProperty("random.int"));
System.out.println(env.getProperty("random.int"));
System.out.println(env.getProperty("random.int"));
System.out.println(env.getProperty("random.uuid"));
System.out.println(env.getProperty("random.uuid"));
System.out.println(env.getProperty("random.uuid"));
}
}6、綁定 spring.main 到 SpringApplication 對(duì)象(*)
把配置文件中的值賦給SpringApplication的默認(rèn)屬性值
public class Step6 {
// 綁定 spring.main 前綴的 key value 至 SpringApplication, 請(qǐng)通過(guò) debug 查看
public static void main(String[] args) throws IOException {
SpringApplication application = new SpringApplication();
ApplicationEnvironment env = new ApplicationEnvironment();
env.getPropertySources().addLast(new ResourcePropertySource("step6", new ClassPathResource("step6.properties")));
System.out.println(application);
Binder.get(env).bind("spring.main", Bindable.ofInstance(application));
System.out.println(application);
}7、打印 banner(*)
public class Step7 {
public static void main(String[] args) {
ApplicationEnvironment env = new ApplicationEnvironment();
SpringApplicationBannerPrinter printer = new SpringApplicationBannerPrinter(
new DefaultResourceLoader(),
new SpringBootBanner()
);
// 測(cè)試文字 banner
// env.getPropertySources().addLast(new MapPropertySource("custom", Map.of("spring.banner.location","banner1.txt")));
// 測(cè)試圖片 banner
// env.getPropertySources().addLast(new MapPropertySource("custom", Map.of("spring.banner.image.location","banner2.png")));
// 版本號(hào)的獲取
System.out.println(SpringBootVersion.getVersion());
printer.print(env, Step7.class, System.out);
}
}8、創(chuàng)建容器
private static GenericApplicationContext createApplicationContext(WebApplicationType type) {
GenericApplicationContext context = null;
switch (type) {
case SERVLET -> context = new AnnotationConfigServletWebServerApplicationContext();
case REACTIVE -> context = new AnnotationConfigReactiveWebServerApplicationContext();
case NONE -> context = new AnnotationConfigApplicationContext();
}
return context;
}9、準(zhǔn)備容器發(fā)布
application context 已初始化事件
10、加載 bean 定義
發(fā)布 application prepared 事件
DefaultListableBeanFactory beanFactory = context.getDefaultListableBeanFactory();
AnnotatedBeanDefinitionReader reader1 = new AnnotatedBeanDefinitionReader(beanFactory);
XmlBeanDefinitionReader reader2 = new XmlBeanDefinitionReader(beanFactory);
ClassPathBeanDefinitionScanner scanner = new ClassPathBeanDefinitionScanner(beanFactory);
reader1.register(Config.class);
reader2.loadBeanDefinitions(new ClassPathResource("b03.xml"));
scanner.scan("com.itheima.a39.sub");11、refresh 容器
發(fā)布 application started 事件
12、執(zhí)行 runner
- 發(fā)布 application ready 事件
- 這其中有異常,發(fā)布 application failed 事件
到此這篇關(guān)于Springboot詳解底層啟動(dòng)過(guò)程的文章就介紹到這了,更多相關(guān)Springboot啟動(dòng)過(guò)程內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot如何通過(guò)Feign調(diào)用傳遞Header中參數(shù)
這篇文章主要介紹了SpringBoot通過(guò)Feign調(diào)用傳遞Header中參數(shù),本文給大家分享兩種解決方案給大家詳細(xì)講解,需要的朋友可以參考下2023-04-04
idea遠(yuǎn)程debug調(diào)試部署在tomcat上項(xiàng)目
本文主要介紹了idea遠(yuǎn)程debug調(diào)試部署在tomcat上項(xiàng)目,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-08-08
復(fù)雜JSON字符串轉(zhuǎn)換為Java嵌套對(duì)象的實(shí)現(xiàn)
這篇文章主要介紹了復(fù)雜JSON字符串轉(zhuǎn)換為Java嵌套對(duì)象的實(shí)現(xiàn),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09
SpringBoot實(shí)現(xiàn)發(fā)送電子郵件
這篇文章主要介紹了SpringBoot實(shí)現(xiàn)發(fā)送電子郵件,電子郵件是—種用電子手段提供信息交換的通信方式,是互聯(lián)網(wǎng)應(yīng)用最廣的服務(wù)。通過(guò)網(wǎng)絡(luò)的電子郵件系統(tǒng),用戶可以非常快速的方式,與世界上任何一個(gè)角落的網(wǎng)絡(luò)用戶聯(lián)系,下面就來(lái)看看SpringBoot如何實(shí)現(xiàn)發(fā)送電子郵件吧2022-01-01
Struts2數(shù)據(jù)輸入驗(yàn)證教程詳解
這篇文章主要介紹了Struts2數(shù)據(jù)輸入驗(yàn)證教程詳解的相關(guān)資料,輸入數(shù)據(jù)驗(yàn)證的方法有兩種,本文給大家介紹的非常詳細(xì),需要的朋友可以參考下2016-10-10
SpringCloudGateway開(kāi)發(fā)過(guò)程解析
這篇文章主要介紹了SpringCloudGateway開(kāi)發(fā)過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-12-12

