Spring Boot通常有一个名为*Application的入口类,在入口类里有一个main方法,这个main方法其实就是一个标准的java应用的入口方法。在main方法中使用SpringApplication.run方法启动SpringBoot应用项目。
@SpringBootApplication是Spring Boot的核心注解。它是一个组合注解,包含了一下三个注解:
如果不使用@SpringBootApplication注解,直接使用@Configuration、@EnableAutoConfiguration、@ComponentScan也能达到相同效果。
是做类似于spring xml 工作的注解,标注在类上,类似与以前的**.xml配置文件。
SpringBoot自动装配需要的注解,会让SpringBoot根据类路径中的jar包依赖为当前项目进行自动装配。同时,它也是一个组合注解,包含:
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@AutoConfigurationPackage
@Import({AutoConfigurationImportSelector.class})
public @interface EnableAutoConfiguration {
String ENABLED_OVERRIDE_PROPERTY = "spring.boot.enableautoconfiguration";
Class<?>[] exclude() default {};
String[] excludeName() default {};
}
其中@Import注解导入了EnableAutoConfigurationImportSelector类,这类就是自动装配关键。
自动装配是SpringBoot的一大特色。
自动配置举例:
添加了spring-boot-starter-web依赖,会自动添加Tomcat和SpringMVC的依赖,SpringBoot会对Tomcat和SpringMVC进行自动配置。
添加了spring-boot-starter-data-jpa依赖,SpringBoot会自动进行JPA相关的配置。
@ComponentScan注解用于告诉Spring哪个packages的用注解标识的类,会被spring自动扫描并且装入bean容器。
SpringBoot默认会自动扫描@SpringBootApplication所在类的同级包以及下级包的Bean(如果为JPA项目还可以扫描标注@Entity的实体类),所以建议入口类放置在最外层包下。
如果需指定扫描路径,需在启动类上加@ComponentScan注解,如下
@SpringCloudApplication
@ComponentScan(value = "com.joker")
public class CoreApplication {
public static void main(String[] args) {
SpringApplication.run(CoreApplication.class, args);
}
}
springboot版本:2.2.2.RELEASE
springboot的启动过程包括:
public SpringApplication(ResourceLoader resourceLoader, Class<?>... primarySources) {
// 初始化资源加载器,默认为null
this.resourceLoader = resourceLoader;
// 断言主要加载资源类不能为空
Assert.notNull(primarySources, "PrimarySources must not be null");
// 初始化主要资源加载类集合并去重
this.primarySources = new LinkedHashSet<>(Arrays.asList(primarySources));
// 得到当前WEB应用类型
this.webApplicationType = WebApplicationType.deduceFromClasspath();
// 设置应用上下文初始化器并去重
setInitializers((Collection) getSpringFactoriesInstances(ApplicationContextInitializer.class));
// 设置监听器并去重
setListeners((Collection) getSpringFactoriesInstances(ApplicationListener.class));
// 得到主应用程序启动类
this.mainApplicationClass = deduceMainApplicationClass();
}
默认为null
一共有三种类型:NONE、SERVLET、REACTIVE
从"META-INF/spring.factories"文件读取key为ApplicationContextInitializer的实例名称集合并去重
从"META-INF/spring.factories"文件读取key为ApplicationListener的实例名称集合并去重(共有11个监听器)
private Class<?> deduceMainApplicationClass() {
try {
StackTraceElement[] stackTrace = new RuntimeException().getStackTrace();
for (StackTraceElement stackTraceElement : stackTrace) {
if ("main".equals(stackTraceElement.getMethodName())) {
return Class.forName(stackTraceElement.getClassName());
}
}
}
catch (ClassNotFoundException ex) {
// Swallow and continue
}
return null;
}
run方法源码如下:
public ConfigurableApplicationContext run(String... args) {
// 创建计时器(任务执行观察者)
StopWatch stopWatch = new StopWatch();
// 开始计时
stopWatch.start();
ConfigurableApplicationContext context = null;
Collection<SpringBootExceptionReporter> exceptionReporters = new ArrayList<>();
// 配置系统属性(Headless)
configureHeadlessProperty();
// 注册监听器
SpringApplicationRunListeners listeners = getRunListeners(args);
// 调用监听方法starting
listeners.starting();
try {
// 加载命令行的参数值
ApplicationArguments applicationArguments = new DefaultApplicationArguments(args);
// 创建并配置应用环境
ConfigurableEnvironment environment = prepareEnvironment(listeners, applicationArguments);
// 配置忽略Bean信息
configureIgnoreBeanInfo(environment);
// 打印banner图形
Banner printedBanner = printBanner(environment);
// 创建应用上下文
context = createApplicationContext();
// 获取异常报告器
exceptionReporters = getSpringFactoriesInstances(SpringBootExceptionReporter.class,
new Class[] { ConfigurableApplicationContext.class }, context);
// 准备应用上下文
prepareContext(context, environment, listeners, applicationArguments, printedBanner);
// 刷新应用上下文
refreshContext(context);
// 上下文刷新完成之后的操作
afterRefresh(context, applicationArguments);
// 结束计时
stopWatch.stop();
if (this.logStartupInfo) {
new StartupInfoLogger(this.mainApplicationClass).logStarted(getApplicationLog(), stopWatch);
}
// 调用监听方法started
listeners.started(context);
// 执行自定义初始化
callRunners(context, applicationArguments);
}
catch (Throwable ex) {
// 异常处理
handleRunFailure(context, ex, exceptionReporters, listeners);
throw new IllegalStateException(ex);
}
try {
// 调用监听方法running
listeners.running(context);
}
catch (Throwable ex) {
// 异常处理
handleRunFailure(context, ex, exceptionReporters, null);
throw new IllegalStateException(ex);
}
return context;
}
创建一个StopWatch实例并执行start方法,这个类主要记录任务的执行时间
配置系统属性java.awt.headless,意义不大,可以忽略。
Headless模式是在缺少显示屏、键盘或者鼠标时候的系统配置。
得到SpringApplicationRunListeners,内部持有:
Log日志类
SpringApplicationRunListener集合(里面只有一个监听器EventPublishingRunListener)
SpringApplicationRunListener是一个接口,EventPublishingRunListener是接口的唯一实现类。
EventPublishingRunListener(事件发布运行监听器) 内部包含三个属性:
private final SpringApplication application;
private final String[] args;
private final SimpleApplicationEventMulticaster initialMulticaster;
该监听器内含事件广播器SimpleApplicationEventMulticaster,主要用于广播(发布)事件。
用于监听SpringApplication的run方法的执行,它有7种方法:
监听器方法 | 调用时间 | 广播的事件 |
---|---|---|
starting | run方法执行的时候立马调用 | ApplicationStartingEvent |
environmentPrepared | ApplicationContext创建之前并且环境信息准备好的时候调用 | ApplicationEnvironmentPreparedEvent |
contextPrepared | ApplicationContext创建好并且在source加载之前调用 | ApplicationContextInitializedEvent |
contextLoaded | ApplicationContext创建并加载之后并在刷新之前调用 | ApplicationPreparedEvent |
started | ApplicationContext已刷新且应用程序已启动,但自定义初始化信息尚未被调用时调用 | ApplicationStartedEvent |
running | ApplicationContext已刷新并且所有自定义初始化信息都已被调用时调用 | ApplicationReadyEvent |
failed | ApplicationContext创建过程中发生故障时调用 | ApplicationFailedEvent |
调用监听方法starting,广播ApplicationStartingEvent事件,触发相关的监听器(4个)
解析在命令行中通过key=value输入的属性值,把他们封装到一个DefaultApplicationArguments类中
创建用程序的环境Environment,并设置比如:环境信息,系统熟悉,输入参数和profile信息。
private ConfigurableEnvironment prepareEnvironment(SpringApplicationRunListeners listeners,
ApplicationArguments applicationArguments) {
// Create and configure the environment
ConfigurableEnvironment environment = getOrCreateEnvironment();
configureEnvironment(environment, applicationArguments.getSourceArgs());
ConfigurationPropertySources.attach(environment);
listeners.environmentPrepared(environment);
bindToSpringApplication(environment);
if (!this.isCustomEnvironment) {
environment = new EnvironmentConverter(getClassLoader()).convertEnvironmentIfNecessary(environment,
deduceEnvironmentClass());
}
ConfigurationPropertySources.attach(environment);
return environment;
}
创建环境对象
创建环境对象,配置4个属性资源:servletConfigInitParams、servletContextInitParams、systemProperties、systemEnvironment
配置环境
调用监听方法environmentPrepared
调用监听方法environmentPrepared,广播ApplicationEnvironmentPreparedEvent事件,触发相关的监听器(7个)
绑定环境到SpringApplication中
也可叫创建Spring容器。根据WebApplicationType来创建应用上下文对象。在构造方法中初始化AnnotatedBeanDefinitionReader和ClassPathBeanDefinitionScanner。
BeanDefinition扫描器,主要作用:扫描带有bean候选者相关注解的类,生成对应的BeanDefinition
bean候选者相关注解:
BeanDefinition读取器,主要作用:
注册内置BeanPostProcessor
注册相关BeanDefinition
在文件META-INF\spring.factories中获取异常报告器集合(里面只有一个报告器:FailureAnalyzers)。
也可叫 准备(配置)Spring容器。
主要步骤如下:
为应用上下文设置环境信息environment
为应用上下文设置转换服务ConversionService
应用上下文初始化
执行应用上下文的初始化。一共有7个初始化器,对应用上下文进行初始化
这些初始化器主要作用是为上下文注入BeanFactoryPostProcessor集合、ApplicationListener集合等
调用监听方法contextPrepared
调用监听方法contextPrepared,广播ApplicationContextInitializedEvent事件,触发相关的监听器(2个)
把SpringApplicationArguments、SpringBootBanner以单例形式注册到BeanFactory中
加载资源配置,完成自动装配
这里只是把资源类(这里只有@SpringBootApplication注解修饰的启动类)注册到BeanFactory中。
调用监听方法contextLoaded
调用监听方法contextLoaded,广播ApplicationPreparedEvent事件,触发相关的监听器(5个)
刷新应用上下文。在这里真正加载bean到容器中
。如果是web容器,会在onRefresh方法中创建一个Server并启动。
准备刷新
prepareRefresh()
刷新上下文环境,初始化上下文环境,对系统的环境便利或者系统属性进行准备和校验
获取刷新后的bean工厂
obtainFreshBeanFactory()
获取新的beanFactory,销毁原有beanFactory、为每个bean生成BeanDefinition等。
注意此处是获取新的,销毁旧的,这就是刷新的意义(Spring容器里通过BeanDefinition对象来表示Bean,BeanDefinition描述了Bean的配置信息。)
准备bean工厂
prepareBeanFactory(beanFactory)
后处理bean工厂
postProcessBeanFactory(beanFactory)
允许在上下文子类中对 bean 工厂进行后置处理。
在beanfactory加载完成所有的bean后,想修改其中某个bean的定义,或者对beanFactory做一些其他的配置,就可以在子类中对beanFactory进行后置处理(子类通过实现接口BeanFactoryPostProcessor来自定义后置处理)。
后置处理器执行的时间是:在所有的beanDenifition加载完成之后,bean实例化之前执行。
调用bean工厂后置处理器
invokeBeanFactoryPostProcessors(beanFactory)
调用在上下文中注册为bean的工厂处理器,执行方法:
注册bean后置处理器
registerBeanPostProcessors(beanFactory)
注册实现了BeanPostProcessor接口的bean。
例如:
初始化消息源
initMessageSource()
初始化国际化工具类MessageSource。
初始化应用事件广播器
initApplicationEventMulticaster()
刷新(初始化特殊的bean)
onRefresh()
这个方法在AbstractApplicationContext中没有实现,留给子类来初始化其他的Bean,是个模板方法,在容器刷新的时候可以自定义逻辑(子类自己去实现逻辑),不同的Spring容器做不同的事情。
SpringBoot项目在这个阶段完成:启动内置Web容器(如Tomcat),并初始化一些特殊的bean(如DataSource)。
这阶段主要完成的事情:
注册监听器
registerListeners()
注册监听器,并且广播earlyApplicationEvents,也就是早期的事件
完成bean工厂初始化
finishBeanFactoryInitialization(beanFactory)
该方法会初始化所有剩余的非懒加载单例bean,并调用BeanPostProcessors。
除了一些内部的 bean、实现了BeanFactoryPostProcessor 接口的 bean、实现了BeanPostProcessor 接口的 bean,其他的非懒加载单例bean都会在这个方法中被初始化,BeanPostProcessor的触发也是在这个方法中。
完成刷新
finishRefresh()
通知生命周期处理器LifecycleProcessor完成刷新过程,同时发出ContextRefreshEvent通知别人
refresh做完之后需要做的其他事情:
上下文刷新完成之后的操作,调用afterRefresh方法。默认什么都不做,主要方便扩展。
广播ApplicationStartedEvent事件。Spring启动结束,开始所有的监听器对象
加载自定义初始化信息,用于扩展,相当于开机启动。
此时会获取到ApplicationRunner、CommandLineRunner这两个接口类型的所有bean集合,循环调用每一个bean的run方法。
广播ApplicationReadyEvent事件。应用启动完成,可以对外提供服务了。