一 、面试提问
1.1 Spring Boot启动流程
面试官:说说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的核心注解,它其实是一个组合注解,包括三个注解
1、@Configuration
: 被标注的类等于在Spring的XML配置文件中(applicationContext.xml),装配所有bean事务,提供了一个Spring的上下文环境
2、@EnableAutoConfiguration
: SpringBoot根据应用所声明的依赖来对Spring框架进行自动配置,其中包括两个重要注解:
-
@AutoConfigurationPackage
:该注解上有一个@Import({Registrar.class})
注解,其中Registrar类的作用是将启动类所在的包下的所有子包组件扫描注入到spring容器中。
因此这就是为什么将controller、service等包放在启动类的同级目录下的原因
-
@Import({AutoConfigurationImportSelector.class})
:其中AutoConfigurationImportSelector
类中有一个getCandidateConfigurations()
方法,这个方法通过SpringFactoriesLoader.loadFactoryNames()
查找位于META-INF/spring.factories
文件中的所有自动配置类并加载这些类。
3、@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);
}
2.在SpringApplication
的构造方法里为SpringApplication
对象赋一些初值。
3.构造方法执行完后,回到run()
方法
该方法中实现了如下几个关键步骤:
-
创建了应用的监听器
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做了一些基本的收尾工作,返回了应用环境上下文。回顾整体流程,Springboot的启动,主要创建了配置环境(environment
)、事件监听(listeners)、应用上下文(applicationContext
),并基于以上条件,在容器中开始实例化我们需要的Bean,至此,通过SpringBoot启动的程序已经构造完成。