一 面试提问

面试官:说说SpringBoot启动流程吧

我 :

首先从main找到run()方法,在执行run()方法之前new一个SpringApplication对象

进入run()方法,创建应用监听器
SpringApplicationRunListeners开始监听

然后加载SpringBoot配置环境(ConfigurableEnvironment),然后把配置环境(Environment)加入监听对象中

然后加载应用上下文(
ConfigurableApplicationContext),当做run方法的返回对象

最后创建Spring容器,refreshContext(context),实现starter自动化配置和bean的实例化等工作。

1.2 SpringBoot自动装配

面试官:说说SpringBoot自动配置的原理

我:通过@EnableAutoConfiguration注解在类路径的META-INF/spring.factories文件中找到所有的对应配置类,然后将这些自动配置类加载到spring容器中

二 知识点详解

2.1 SpringBoot核心注解:

@SpringBootApplication注解是Spring Boot的核心注解,它其实是一个组合注解,包括三个注解

@Configuration:被标注的类等于在Spring的XML配置文件中(applicationContext.xml),装配所有bean事务,提供了一个Spring的上下文环境

@
EnableAutoConfiguration:SpringBoot根据应用所声明的依赖来对Spring框架进行自动配置,其中包括两个重要注解:

@AutoConfigurationPackage:该注解上有一个@Import({Registrar.class})注解,其中Registrar类的作用是将启动类所在的包下的所有子包组件扫描注入到spring容器中。

因此这就是为什么将controller、service等包放在启动类的同级目录下的原因

@Import({
AutoConfigurationImportSelector.class}):其中AutoConfigurationImportSelector类中有一个getCandidateConfigurations()方法,这个方法通过SpringFactoriesLoader.loadFactoryNames()查找位于META-INF/spring.factories文件中的所有自动配置类并加载这些类。

springboot 生产环境指定yml 生产环境springboot如何启动_spring

 

@ComponentScan:组件扫描,自动扫描和装配Bean,扫描SpringApplication的run方法中的
ExammanagerApplication.class所在的包路径下的文件,因此将启动类(main)放在跟包路径下。它去寻找带有@Component注解的类,并为其创建bean。

2.2详细启动流程(结合源码)

  1. 首先进入run()方法,run方法中new创建了一个SpringApplication实例
public static ConfigurableApplicationContext run(Class<?>[] primarySources, String[] args) {
        return (new SpringApplication(primarySources)).run(args);
    }

在SpringApplication的构造方法里为SpringApplication对象赋一些初值。

构造方法执行完后,回到run()方法

springboot 生产环境指定yml 生产环境springboot如何启动_mybatis_02

 

该方法中实现了如下几个关键步骤:

创建了应用的监听器
SpringApplicationRunListeners并开始监听

加载SpringBoot配置环境(ConfigurableEnvironment),如果是通过web容器发布,会加载StandardEnvironment,其最终也是继承了ConfigurableEnvironment

配置环境(Environment)加入到监听器对象中(
SpringApplicationRunListeners)

创建run方法的返回对象:
ConfigurableApplicationContext(应用配置上下文)

回到run方法内,prepareContext方法将listeners、environment、applicationArguments、banner等重要组件与上下文对象关联

接下来的refreshContext(context)方法(初始化方法如下)将是实现spring-boot-starter-*(mybatis、redis等)自动化配置的关键,包括spring.factories的加载,bean的实例化等核心工作。

springboot 生产环境指定yml 生产环境springboot如何启动_mybatis_03

 

配置结束后,Springboot做了一些基本的收尾工作,返回了应用环境上下文。回顾整体流程,Springboot的启动,主要创建了配置环境(environment)、事件监听(listeners)、应用上下文(applicationContext),并基于以上条件,在容器中开始实例化我们需要的Bean,至此,通过SpringBoot启动的程序已经构造完成。