Springboot启动配置原理

1Springboot启动配置原理

几个重要的事件回调机制(配置在META-INF/spring.factories)

ApplicationContextInitializer
SpringApplicationRunListener

只需要放在ioc容器中

ApplicationRunner
CommandLineRunner

1.1启动流程

initialize(sources);
private void initialize(Object[] sources) {
//保存主配置类
if (sources != null && sources.length > 0) {
this.sources.addAll(Arrays.asList(sources));
}
//判断当前是否一个web应用
this.webEnvironment = deduceWebEnvironment();
//从类路径下找到META‐INF/spring.factories配置的所有ApplicationContextInitializer;然后保存起
来
setInitializers((Collection) getSpringFactoriesInstances(
ApplicationContextInitializer.class));
//从类路径下找到ETA‐INF/spring.factories配置的所有ApplicationListener
setListeners((Collection) getSpringFactoriesInstances(ApplicationListener.class));
//从多个配置类中找到有main方法的主配置类
this.mainApplicationClass = deduceMainApplicationClass();
}

//从类路径下找到META‐INF/spring.factories配置的所有ApplicationContextInitializer;涉及的代码

1.2、运行run方法

public ConfigurableApplicationContext run(String... args) {
StopWatch stopWatch = new StopWatch();
stopWatch.start();

ConfigurableApplicationContext context = null;
FailureAnalyzers analyzers = null;
configureHeadlessProperty();
//获取SpringApplicationRunListeners;从类路径下META‐INF/spring.factories
SpringApplicationRunListeners listeners = getRunListeners(args);
//回调所有的获取SpringApplicationRunListener.starting()方法
listeners.starting();
try {
//封装命令行参数
ApplicationArguments applicationArguments = new DefaultApplicationArguments(
args);
//准备环境
ConfigurableEnvironment environment = prepareEnvironment(listeners,
applicationArguments);
//创建环境完成后回调SpringApplicationRunListener.environmentPrepared();表示环境准
备完成
Banner printedBanner = printBanner(environment);
//创建ApplicationContext;决定创建web的ioc还是普通的ioc
context = createApplicationContext();
analyzers = new FailureAnalyzers(context);
//准备上下文环境;将environment保存到ioc中;而且applyInitializers();
//applyInitializers():回调之前保存的所有的ApplicationContextInitializer的initialize方法
//回调所有的SpringApplicationRunListener的contextPrepared();
//
prepareContext(context, environment, listeners, applicationArguments,
printedBanner);
//prepareContext运行完成以后回调所有的SpringApplicationRunListener的contextLoaded();
//s刷新容器;ioc容器初始化(如果是web应用还会创建嵌入式的Tomcat);Spring注解版
//扫描,创建,加载所有组件的地方;(配置类,组件,自动配置)
refreshContext(context);
//从ioc容器中获取所有的ApplicationRunner和CommandLineRunner进行回调
//ApplicationRunner先回调,CommandLineRunner再回调
afterRefresh(context, applicationArguments);
//所有的SpringApplicationRunListener回调finished方法
listeners.finished(context, null);
stopWatch.stop();
if (this.logStartupInfo) {
new StartupInfoLogger(this.mainApplicationClass)
.logStarted(getApplicationLog(), stopWatch);
}
//整个SpringBoot应用启动完成以后返回启动的ioc容器;
return context;
}
catch (Throwable ex) {
handleRunFailure(context, listeners, analyzers, ex);
throw new IllegalStateException(ex);
}
}

1.3、事件监听机制(验证启动过程)

配置在META-INF/spring.factories

ApplicationContextInitializer


public class HelloApplicationContextInitializer implements ApplicationContextInitializer<ConfigurableApplicationContext> {
    @Override
    public void initialize(ConfigurableApplicationContext applicationContext) {
        System.out.println("ApplicationContextInitializer执行initialize方法:参数"+applicationContext);
    }
}

SpringApplicationRunListener

package com.atguigu.springboot.listener;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.SpringApplicationRunListener;
import org.springframework.context.ConfigurableApplicationContext;
import org.springframework.core.env.ConfigurableEnvironment;

public class HelloSpringApplicationRunListener implements SpringApplicationRunListener {

    //必须有的构造器
    public HelloSpringApplicationRunListener(SpringApplication application, String[] args){

    }

    @Override
    public void starting() {
        System.out.println("SpringApplicationRunListener   starting方法");
    }

    @Override
    public void environmentPrepared(ConfigurableEnvironment environment) {
        System.out.println("SpringApplicationRunListener   environmentPrepared 方法");
    }

    @Override
    public void contextPrepared(ConfigurableApplicationContext context) {
        System.out.println("SpringApplicationRunListener   contextPrepared 方法");
    }

    @Override
    public void contextLoaded(ConfigurableApplicationContext context) {
        System.out.println("SpringApplicationRunListener    contextLoaded 方法");
    }

    @Override
    public void finished(ConfigurableApplicationContext context, Throwable exception) {
        System.out.println("SpringApplicationRunListener    finished 方法");
    }
}

配置(META-INF/spring.factories)

 

org.springframework.context.ApplicationContextInitializer=\
com.atguigu.springboot.listener.HelloApplicationContextInitializer

org.springframework.boot.SpringApplicationRunListener=\
com.atguigu.springboot.listener.HelloSpringApplicationRunListener

 

只需要放在ioc容器中

ApplicationRunner
@Component
public class HelloApplicationRunner implements ApplicationRunner {
    @Override
    public void run(ApplicationArguments args) throws Exception {
        System.out.println("ApplicationRunner...run....");
    }
}
CommandLineRunner
@Component
public class HelloCommandLineRunner implements CommandLineRunner {
    @Override
    public void run(String... args) throws Exception {
        System.out.println("CommandLineRunner...run..."+ Arrays.asList(args));
    }
}

 

2启动测试

H:\java\jdk1.8.0_181\bin\java -XX:TieredStopAtLevel=1 -noverify -Dspring.output.ansi.enabled=always -Dcom.sun.management.jmxremote -Dcom.sun.management.jmxremote.port=58791 -Dcom.sun.management.jmxremote.authenticate=false -Dcom.sun.management.jmxremote.ssl=false -Dspring.liveBeansView.mbeanDomain -Dspring.application.admin.enabled=true -javaagent:H:\IntelliJ_IDEA\lib\idea_rt.jar=58792:H:\IntelliJ_IDEA\bin -Dfile.encoding=UTF-8 -classpath H:\java\jdk1.8.0_181\jre\lib\charsets.jar;H:\java\jdk1.8.0_181\jre\lib\deploy.jar;H:\java\jdk1.8.0_181\jre\lib\ext\access-bridge-64.jar;H:\java\jdk1.8.0_181\jre\lib\ext\cldrdata.jar;H:\java\jdk1.8.0_181\jre\lib\ext\dnsns.jar;H:\java\jdk1.8.0_181\jre\lib\ext\jaccess.jar;H:\java\jdk1.8.0_181\jre\lib\ext\jfxrt.jar;H:\java\jdk1.8.0_181\jre\lib\ext\localedata.jar;H:\java\jdk1.8.0_181\jre\lib\ext\nashorn.jar;H:\java\jdk1.8.0_181\jre\lib\ext\sunec.jar;H:\java\jdk1.8.0_181\jre\lib\ext\sunjce_provider.jar;H:\java\jdk1.8.0_181\jre\lib\ext\sunmscapi.jar;H:\java\jdk1.8.0_181\jre\lib\ext\sunpkcs11.jar;H:\java\jdk1.8.0_181\jre\lib\ext\zipfs.jar;H:\java\jdk1.8.0_181\jre\lib\javaws.jar;H:\java\jdk1.8.0_181\jre\lib\jce.jar;H:\java\jdk1.8.0_181\jre\lib\jfr.jar;H:\java\jdk1.8.0_181\jre\lib\jfxswt.jar;H:\java\jdk1.8.0_181\jre\lib\jsse.jar;H:\java\jdk1.8.0_181\jre\lib\management-agent.jar;H:\java\jdk1.8.0_181\jre\lib\plugin.jar;H:\java\jdk1.8.0_181\jre\lib\resources.jar;H:\java\jdk1.8.0_181\jre\lib\rt.jar;H:\developSource\IDEA_workSpace\代码\spring-boot-07\target\classes;D:\devsoft\apache-maven-3.3.9\respository\org\springframework\boot\spring-boot-starter-web\1.5.10.RELEASE\spring-boot-starter-web-1.5.10.RELEASE.jar;D:\devsoft\apache-maven-3.3.9\respository\org\springframework\boot\spring-boot-starter\1.5.10.RELEASE\spring-boot-starter-1.5.10.RELEASE.jar;D:\devsoft\apache-maven-3.3.9\respository\org\springframework\boot\spring-boot\1.5.10.RELEASE\spring-boot-1.5.10.RELEASE.jar;D:\devsoft\apache-maven-3.3.9\respository\org\springframework\boot\spring-boot-autoconfigure\1.5.10.RELEASE\spring-boot-autoconfigure-1.5.10.RELEASE.jar;D:\devsoft\apache-maven-3.3.9\respository\org\springframework\boot\spring-boot-starter-logging\1.5.10.RELEASE\spring-boot-starter-logging-1.5.10.RELEASE.jar;D:\devsoft\apache-maven-3.3.9\respository\ch\qos\logback\logback-classic\1.1.11\logback-classic-1.1.11.jar;D:\devsoft\apache-maven-3.3.9\respository\ch\qos\logback\logback-core\1.1.11\logback-core-1.1.11.jar;D:\devsoft\apache-maven-3.3.9\respository\org\slf4j\jcl-over-slf4j\1.7.25\jcl-over-slf4j-1.7.25.jar;D:\devsoft\apache-maven-3.3.9\respository\org\slf4j\jul-to-slf4j\1.7.25\jul-to-slf4j-1.7.25.jar;D:\devsoft\apache-maven-3.3.9\respository\org\slf4j\log4j-over-slf4j\1.7.25\log4j-over-slf4j-1.7.25.jar;D:\devsoft\apache-maven-3.3.9\respository\org\yaml\snakeyaml\1.17\snakeyaml-1.17.jar;D:\devsoft\apache-maven-3.3.9\respository\org\springframework\boot\spring-boot-starter-tomcat\1.5.10.RELEASE\spring-boot-starter-tomcat-1.5.10.RELEASE.jar;D:\devsoft\apache-maven-3.3.9\respository\org\apache\tomcat\embed\tomcat-embed-core\8.5.27\tomcat-embed-core-8.5.27.jar;D:\devsoft\apache-maven-3.3.9\respository\org\apache\tomcat\tomcat-annotations-api\8.5.27\tomcat-annotations-api-8.5.27.jar;D:\devsoft\apache-maven-3.3.9\respository\org\apache\tomcat\embed\tomcat-embed-el\8.5.27\tomcat-embed-el-8.5.27.jar;D:\devsoft\apache-maven-3.3.9\respository\org\apache\tomcat\embed\tomcat-embed-websocket\8.5.27\tomcat-embed-websocket-8.5.27.jar;D:\devsoft\apache-maven-3.3.9\respository\org\hibernate\hibernate-validator\5.3.6.Final\hibernate-validator-5.3.6.Final.jar;D:\devsoft\apache-maven-3.3.9\respository\javax\validation\validation-api\1.1.0.Final\validation-api-1.1.0.Final.jar;D:\devsoft\apache-maven-3.3.9\respository\org\jboss\logging\jboss-logging\3.3.1.Final\jboss-logging-3.3.1.Final.jar;D:\devsoft\apache-maven-3.3.9\respository\com\fasterxml\classmate\1.3.4\classmate-1.3.4.jar;D:\devsoft\apache-maven-3.3.9\respository\com\fasterxml\jackson\core\jackson-databind\2.8.10\jackson-databind-2.8.10.jar;D:\devsoft\apache-maven-3.3.9\respository\com\fasterxml\jackson\core\jackson-annotations\2.8.0\jackson-annotations-2.8.0.jar;D:\devsoft\apache-maven-3.3.9\respository\com\fasterxml\jackson\core\jackson-core\2.8.10\jackson-core-2.8.10.jar;D:\devsoft\apache-maven-3.3.9\respository\org\springframework\spring-web\4.3.14.RELEASE\spring-web-4.3.14.RELEASE.jar;D:\devsoft\apache-maven-3.3.9\respository\org\springframework\spring-aop\4.3.14.RELEASE\spring-aop-4.3.14.RELEASE.jar;D:\devsoft\apache-maven-3.3.9\respository\org\springframework\spring-beans\4.3.14.RELEASE\spring-beans-4.3.14.RELEASE.jar;D:\devsoft\apache-maven-3.3.9\respository\org\springframework\spring-context\4.3.14.RELEASE\spring-context-4.3.14.RELEASE.jar;D:\devsoft\apache-maven-3.3.9\respository\org\springframework\spring-webmvc\4.3.14.RELEASE\spring-webmvc-4.3.14.RELEASE.jar;D:\devsoft\apache-maven-3.3.9\respository\org\springframework\spring-expression\4.3.14.RELEASE\spring-expression-4.3.14.RELEASE.jar;D:\devsoft\apache-maven-3.3.9\respository\org\slf4j\slf4j-api\1.7.25\slf4j-api-1.7.25.jar;D:\devsoft\apache-maven-3.3.9\respository\org\springframework\spring-core\4.3.14.RELEASE\spring-core-4.3.14.RELEASE.jar com.atguigu.springboot.SpringBoot07Application
SpringApplicationRunListener   starting方法
SpringApplicationRunListener   environmentPrepared 方法

  .   ____          _            __ _ _
 /\\ / ___'_ __ _ _(_)_ __  __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
 \\/  ___)| |_)| | | | | || (_| |  ) ) ) )
  '  |____| .__|_| |_|_| |_\__, | / / / /
 =========|_|==============|___/=/_/_/_/
 :: Spring Boot ::       (v1.5.10.RELEASE)

ApplicationContextInitializer执行initialize方法:参数org.springframework.boot.context.embedded.AnnotationConfigEmbeddedWebApplicationContext@64d7f7e0: startup date [Thu Jan 01 08:00:00 CST 1970]; root of context hierarchy
SpringApplicationRunListener   contextPrepared 方法
2019-06-04 07:38:19.260  INFO 10556 --- [           main] c.a.springboot.SpringBoot07Application   : Starting SpringBoot07Application on zhoufei with PID 10556 (started by Administrator in H:\developSource\IDEA_workSpace\代码\spring-boot-07)
2019-06-04 07:38:19.263  INFO 10556 --- [           main] c.a.springboot.SpringBoot07Application   : No active profile set, falling back to default profiles: default
SpringApplicationRunListener    contextLoaded 方法
2019-06-04 07:38:19.294  INFO 10556 --- [           main] ationConfigEmbeddedWebApplicationContext : Refreshing org.springframework.boot.context.embedded.AnnotationConfigEmbeddedWebApplicationContext@64d7f7e0: startup date [Tue Jun 04 07:38:19 CST 2019]; root of context hierarchy
2019-06-04 07:38:20.075  INFO 10556 --- [           main] s.b.c.e.t.TomcatEmbeddedServletContainer : Tomcat initialized with port(s): 8080 (http)
2019-06-04 07:38:20.082  INFO 10556 --- [           main] o.apache.catalina.core.StandardService   : Starting service [Tomcat]
2019-06-04 07:38:20.083  INFO 10556 --- [           main] org.apache.catalina.core.StandardEngine  : Starting Servlet Engine: Apache Tomcat/8.5.27
2019-06-04 07:38:20.157  INFO 10556 --- [ost-startStop-1] o.a.c.c.C.[Tomcat].[localhost].[/]       : Initializing Spring embedded WebApplicationContext
2019-06-04 07:38:20.157  INFO 10556 --- [ost-startStop-1] o.s.web.context.ContextLoader            : Root WebApplicationContext: initialization completed in 863 ms
2019-06-04 07:38:20.248  INFO 10556 --- [ost-startStop-1] o.s.b.w.servlet.ServletRegistrationBean  : Mapping servlet: 'dispatcherServlet' to [/]
2019-06-04 07:38:20.252  INFO 10556 --- [ost-startStop-1] o.s.b.w.servlet.FilterRegistrationBean   : Mapping filter: 'characterEncodingFilter' to: [/*]
2019-06-04 07:38:20.253  INFO 10556 --- [ost-startStop-1] o.s.b.w.servlet.FilterRegistrationBean   : Mapping filter: 'hiddenHttpMethodFilter' to: [/*]
2019-06-04 07:38:20.253  INFO 10556 --- [ost-startStop-1] o.s.b.w.servlet.FilterRegistrationBean   : Mapping filter: 'httpPutFormContentFilter' to: [/*]
2019-06-04 07:38:20.253  INFO 10556 --- [ost-startStop-1] o.s.b.w.servlet.FilterRegistrationBean   : Mapping filter: 'requestContextFilter' to: [/*]
2019-06-04 07:38:20.609  INFO 10556 --- [           main] s.w.s.m.m.a.RequestMappingHandlerAdapter : Looking for @ControllerAdvice: org.springframework.boot.context.embedded.AnnotationConfigEmbeddedWebApplicationContext@64d7f7e0: startup date [Tue Jun 04 07:38:19 CST 2019]; root of context hierarchy
2019-06-04 07:38:20.694  INFO 10556 --- [           main] s.w.s.m.m.a.RequestMappingHandlerMapping : Mapped "{[/error]}" onto public org.springframework.http.ResponseEntity<java.util.Map<java.lang.String, java.lang.Object>> org.springframework.boot.autoconfigure.web.BasicErrorController.error(javax.servlet.http.HttpServletRequest)
2019-06-04 07:38:20.694  INFO 10556 --- [           main] s.w.s.m.m.a.RequestMappingHandlerMapping : Mapped "{[/error],produces=[text/html]}" onto public org.springframework.web.servlet.ModelAndView org.springframework.boot.autoconfigure.web.BasicErrorController.errorHtml(javax.servlet.http.HttpServletRequest,javax.servlet.http.HttpServletResponse)
2019-06-04 07:38:20.740  INFO 10556 --- [           main] o.s.w.s.handler.SimpleUrlHandlerMapping  : Mapped URL path [/webjars/**] onto handler of type [class org.springframework.web.servlet.resource.ResourceHttpRequestHandler]
2019-06-04 07:38:20.740  INFO 10556 --- [           main] o.s.w.s.handler.SimpleUrlHandlerMapping  : Mapped URL path [/**] onto handler of type [class org.springframework.web.servlet.resource.ResourceHttpRequestHandler]
2019-06-04 07:38:20.784  INFO 10556 --- [           main] o.s.w.s.handler.SimpleUrlHandlerMapping  : Mapped URL path [/**/favicon.ico] onto handler of type [class org.springframework.web.servlet.resource.ResourceHttpRequestHandler]
2019-06-04 07:38:20.942  INFO 10556 --- [           main] o.s.j.e.a.AnnotationMBeanExporter        : Registering beans for JMX exposure on startup
2019-06-04 07:38:21.082  INFO 10556 --- [           main] s.b.c.e.t.TomcatEmbeddedServletContainer : Tomcat started on port(s): 8080 (http)
ApplicationRunner...run....
CommandLineRunner...run...[]
SpringApplicationRunListener    finished 方法
2019-06-04 07:38:21.088  INFO 10556 --- [           main] c.a.springboot.SpringBoot07Application   : Started SpringBoot07Application in 2.098 seconds (JVM running for 3.212)

 

验证代码

 

public class HelloApplicationContextInitializer implements ApplicationContextInitializer<ConfigurableApplicationContext> {
    @Override
    public void initialize(ConfigurableApplicationContext applicationContext) {
        System.out.println("ApplicationContextInitializer执行initialize方法:参数"+applicationContext);
    }
}
-----------------------------
@Component
public class HelloApplicationRunner implements ApplicationRunner {
    @Override
    public void run(ApplicationArguments args) throws Exception {
        System.out.println("ApplicationRunner...run....");
    }
}

-----------------------------

@Component
public class HelloCommandLineRunner implements CommandLineRunner {
    @Override
    public void run(String... args) throws Exception {
        System.out.println("CommandLineRunner...run..."+ Arrays.asList(args));
    }
}

-----------------------------
public class HelloSpringApplicationRunListener implements SpringApplicationRunListener {

    //必须有的构造器
    public HelloSpringApplicationRunListener(SpringApplication application, String[] args){

    }

    @Override
    public void starting() {
        System.out.println("SpringApplicationRunListener   starting方法");
    }

    @Override
    public void environmentPrepared(ConfigurableEnvironment environment) {
        System.out.println("SpringApplicationRunListener   environmentPrepared 方法");
    }

    @Override
    public void contextPrepared(ConfigurableApplicationContext context) {
        System.out.println("SpringApplicationRunListener   contextPrepared 方法");
    }

    @Override
    public void contextLoaded(ConfigurableApplicationContext context) {
        System.out.println("SpringApplicationRunListener    contextLoaded 方法");
    }

    @Override
    public void finished(ConfigurableApplicationContext context, Throwable exception) {
        System.out.println("SpringApplicationRunListener    finished 方法");
    }
}



-----------------------------

@SpringBootApplication
public class SpringBoot07Application {

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



-----------------------------

org.springframework.context.ApplicationContextInitializer=\
com.atguigu.springboot.listener.HelloApplicationContextInitializer

org.springframework.boot.SpringApplicationRunListener=\
com.atguigu.springboot.listener.HelloSpringApplicationRunListener

 

 

 

 

 

 

 

 

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值