主启动类
@SpringBootApplication
@MapperScan("com.example.mapper")
public class StableBootApplication {public static void main(String[] args) {SpringApplication.run(StableBootApplication.class,args);}
}
SpringApplication类中有个静态run()方法,最终执行如下:
public static ConfigurableApplicationContext run(Class<?>[] primarySources, String[] args) {return (new SpringApplication(primarySources)).run(args);}
显然 分为两部分
- 实例化(new) SpringApplication。
- 执行run()方法。
阅读 SpringApplication类的构造函数
实例化 SpringApplication
获取并且设置应用类型(重要)
应用类型有三种:
- NONE:顾名思义,什么都没有,正常流程走,不额外的启动web容器,比如Tomcat。
- SERVLET:基于servlet的web程序,需要启动内嵌的servletweb容器,比如Tomcat。(常用)
- REACTIVE:基于reactive的web程序,需要启动内嵌reactiveweb容器,作者不是很了解,不便多说
源码如下:
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;}
}
获取并且设置初始化器以及监听器
初始化器以及监听器 调用的方法都是 getSpringFactoriesInstances(类型Type)。
最终逻辑是 从 从类路径META-INF/spring.factories中 加载 初始化器以及监听器 。
初始化器(ApplicationContextInitializer): 初始化某些 IOC容器刷新之前的组件。
监听器(ApplicationListener): 监听特定的事件 比如IOC容器刷新、容器关闭等。
private <T> Collection<T> getSpringFactoriesInstances(Class<T> type, Class<?>[] parameterTypes, Object... args) {ClassLoader classLoader = this.getClassLoader();// 从 META-INF/spring.factories 中获取Set<String> names = new LinkedHashSet(SpringFactoriesLoader.loadFactoryNames(type, classLoader));// 根据名称实例化List<T> instances = this.createSpringFactoriesInstances(type, parameterTypes, classLoader, args, names);AnnotationAwareOrderComparator.sort(instances);return instances;
}
小结
SpringApplication的构建都是为了run()方法启动做铺垫,最重要的部分就是设置应用类型、设置初始化器、设置监听器。
注意: 初始化器和这里的监听器都要放置在spring.factories文件中才能在这一步骤加载,否则不会生效。因为此时IOC容器还未创建,所以即使将其注入到IOC容器中也是不会生效的。
执行run() 方法
上面分析了SpringApplication的构建过程,一切都做好了铺垫,现在到了启动的过程了。
获取、启动运行过程监听器
监听器(SpringApplicationRunListeners): 监听应用程序启动过程。
获取 loadFactoryNames()方法从spring.factories文件中获取值:
org.springframework.boot.SpringApplicationRunListener=
org.springframework.boot.context.event.EventPublishingRunListener
启动运行监听器
//执行starting()方法
listeners.starting(bootstrapContext, this.mainApplicationClass);
环境构建
加载系统配置以及用户的自定义配置(application.properties)
ConfigurableEnvironment environment =
this.prepareEnvironment(listeners, bootstrapContext, applicationArguments);
创建IOC容器
context = this.createApplicationContext();
很显然 我这里是 SERVLET 故创建的是 AnnotationConfigServletWebServerApplicationContext()。
这一步仅仅是创建了IOC容器,未有其他操作
刷新IOC容器的前置处理(重要)
this.prepareContext(bootstrapContext, context, environment, listeners,
applicationArguments, printedBanner);
调用初始化器
从spring.factories取值的
将自定义的ApplicationContextInitializer放在META-INF/spring.factories中,在此时也是会被调用。
加载启动类,注入容器
两次广播事件
listeners.contextPrepared(context);
load(context, sources.toArray(new Object[0]));
刷新IOC容器(重要)
刷新容器完全是Spring的功能了,比如初始化资源,初始化上下文广播器等
tomcat内置应用服务器在此时启动
刷新IOC容器的后置处理
this.afterRefresh(context, applicationArguments);
默认为空,如果有自定义需求可以重写,比如打印一些启动结束日志等。
发出结束执行的事件
执行Runners
小结
未完待续…
springboot启动流程详解(清晰易懂)
springboot内置服务器tomca启动详解
spring boot源码解读