data:image/s3,"s3://crabby-images/5a36e/5a36e8d9f24b5d203e922dad0945d71e0cbe0280" alt="Spring Boot实战:从0开始动手搭建企业级项目"
5.4 SpringApplication启动流程解析
Spring Boot项目通过运行启动类中的run()方法就可以将整个应用启动。那么这个方法究竟做了哪些神奇的事情呢?SpringApplication启动流程又做了哪些操作呢?接下来通过源码一探究竟。
点击启动类中的run()方法进入SpringApplication类,源码及注释如下所示:
data:image/s3,"s3://crabby-images/e6748/e6748539d5e6acf979baa4f4c3342b343c6b7164" alt=""
data:image/s3,"s3://crabby-images/c9d13/c9d13c7774f103ecb5208574bb49fb53a55341a6" alt=""
data:image/s3,"s3://crabby-images/54314/54314bd97fc6f03faa87576a6d6d569de05d3623" alt=""
Spring Boot项目启动步骤分析如下所示。
(1)实例化SpringApplication对象。
在执行run()方法前,使用new SpringApplication()构造SpringApplication对象。SpringApplication类的构造方法如下所示:
data:image/s3,"s3://crabby-images/25a8d/25a8d8343fd49b94e9c472edf6d72b5dd041ea82" alt=""
这一步主要是构造SpringApplication对象,并为SpringApplication的属性赋值,在构造完成后,开始执行run()方法。
比较重要的一个知识点是webApplicationType值的设置,其目的是获取当前应用的类型,对后续步骤构造容器环境和Spring容器的初始化起到作用。该值的获取是通过调用WebApplicationType.deduceFromClasspath()方法得到的,该方法源码及注释如下所示:
data:image/s3,"s3://crabby-images/cc7cc/cc7ccde1ae6fed7b54b50e8b1c79eac58b3edd67" alt=""
data:image/s3,"s3://crabby-images/bf390/bf390d50a726169fccbcc168d216de09b163764c" alt=""
data:image/s3,"s3://crabby-images/d2e7c/d2e7cfbd9ae41f0faf070e4dd30ca34b8fe04c52" alt=""
WebApplicationType的值有3个,分别如下所示。
①SERVLET:Servlet环境。
②REACTIVE:Reactive环境。
③NONE:非Web环境。
在deduceFromClasspath()方法中代码多次调用ClassUtils.isPresent()方法,以此判断在常量中的类是否存在。该方法的最终实现原理通过Class.forName加载某个类,如果成功加载,则证明这个类存在,反之则代表该类不存在。
deduceFromClasspath()方法的实现逻辑如下:先判断webflux相关的类是否存在,存在则认为当前应用为REACTIVE类型;不存在则继续判断SERVLET相关的类是否存在,都不存在则为NONE类型;否则,当前应用为SERVLET类型。具体的类加载判断方法可以直接查看源码,相关的代码注释笔者也已经标注在代码中。
以newbee-mall项目举例,由于项目中引用了spring-boot-starter-web且并未引用webflux相关的类,所以newbee-mall项目类型为SERVLET类型。
(2)开始执行run()方法,代码执行时间的监控开启,在Spring Boot应用启动成功后会打印启动时间。
(3)配置headless属性,java.awt.headles是J2SE的一种模式,用于在缺失显示屏、鼠标或者键盘时的系统配置,默认为true。通俗而言,该行代码的作用是Spring Boot应用在启动时,没有检测到显示器也能够继续执行后面的步骤。
(4)获取SpringApplicationRunListeners,getRunListeners()方法的源码如下所示:
data:image/s3,"s3://crabby-images/ea005/ea00513a45c36fb30414ed2517bde78ae6fbc632" alt=""
这里会调用SpringFactoriesLoader类中的loadFactoryNames()方法。该方法在介绍自动配置时已经讲解过,与获取自动配置类的类名相同。也就是在getRunListeners()方法中调用该方法是从类路径META-INF/spring.factories中获取SpringApplication RunListener指定类的。在spring-boot-2.3.7.RELEASE.jar包中的META-INF目录下找到了spring.factories文件,当前文件中只有一个RunListener,即org.springframework. boot.context.event.EventPublishingRunListener,如图5-6所示。
data:image/s3,"s3://crabby-images/b5489/b5489c7803d04643ed823c87e082ba0d964423ce" alt=""
图5-6 META-INF/spring.factories文件
通过debug模式也可以得出该类为org.springframework.boot.context. event.Event PublishingRunListener。在“listeners.starting();”代码前输入一个断点,之后通过debug模式启动项目,可以看出此时加载的listener为EventPublishingRunListener,如图5-7所示。
data:image/s3,"s3://crabby-images/5eb83/5eb836fb7fbf9322924cf9a9bb30a0a89c14b414" alt=""
图5-7 EventPublishingRunListener类
(5)回调SpringApplicationRunListener对象的starting()方法。
(6)解析run()方法的args参数并封装为DefaultApplicationArguments类。
(7)prepareEnvironment()方法的作用与它的方法名的含义相同,就是为当前应用准备一个Environment对象,也就是运行环境。它主要完成对ConfigurableEnvironment的初始化工作。该方法的源码及解析如下所示:
data:image/s3,"s3://crabby-images/d9875/d9875df084e873d48a881d3a23b0e1244466c59d" alt=""
data:image/s3,"s3://crabby-images/d6b7a/d6b7a6d23ec6c424ef7ea64ee72661cc27ac8cb4" alt=""
由于项目中存在spring-boot-starter-web依赖,webApplicationType的值为WebApplicationType.SERVLET,所以getOrCreateEnvironment()方法返回的是StandardServletEnvironment对象,是一个标准的Servlet环境。StandardServletEnvironment是整个Spring Boot项目运行环境的实现类,后续关于环境的设置都基于此类。
在创建环境完成后,接下来是配置环境,configureEnvironment()方法的源码如下所示:
data:image/s3,"s3://crabby-images/d171c/d171caca9fdbdf88863ba2a48d43eb1221d0f781" alt=""
该方法主要加载一些默认配置,在执行完这一步骤后,会触发监听器(主要触发ConfigFileApplicationListener),将会加载application.properties或者application.yml配置文件。
(8)设置系统参数,configureIgnoreBeanInfo()方法的源码如下所示:
data:image/s3,"s3://crabby-images/3c4d9/3c4d93aea03f632cfc6c11d0f83c7e81df322f08" alt=""
查看源码可知,该方法会获取spring.beaninfo.ignore配置项的值,即使未获取也没有关系。代码的最后还是给该配置项输入了一个默认值true,表示跳过对BeanInfo类的搜索,它无特别含义,不用深究该步骤。
(9)获取需要打印的Spring Boot启动Banner对象,源码如下所示:
data:image/s3,"s3://crabby-images/1535f/1535fea006802694ba239cedcfbf3ec88c7cab22" alt=""
首先判断当前是否允许打印Banner,默认会打印到控制台上,之后获取Banner对象。而Spring Boot目前支持图片Banner和文字Banner,如果开发人员做了Banner配置则会在控制台打印开发人员配置的Banner,否则打印默认Banner。默认Banner的实现类为org.springframework.boot.SpringBootBanner,源码如下所示:
data:image/s3,"s3://crabby-images/7eaf0/7eaf07e20c172f4f82e1360610c82ef568ccd9ad" alt=""
data:image/s3,"s3://crabby-images/1fdb4/1fdb4a6e2b65e4476904332a1197653b6b16c33e" alt=""
BANNER变量就是在默认情况下打印在控制台上的Banner。而printBanner()方法,就是把定义好的Banner和Spring Boot的版本号打印出来。
其实在Banner打印流程中也能够看出Spring Boot框架约定优于配置的特性。开发人员配置Banner就使用开发人员配置的,如果没有,就使用Spring Boot默认的。
Spring Boot框架的约定优于配置理念正是“你配置就用你配置的,你不配置就用约定好的”。
(10)创建Spring容器ApplicationContext,createApplicationContext()方法的源码如下所示:
data:image/s3,"s3://crabby-images/30ba3/30ba3818d262ea90c30e98df72fb450d3c29e50e" alt=""
data:image/s3,"s3://crabby-images/ba2cb/ba2cb24d0bfa8d824248bb341a213557ecc2cd2c" alt=""
通过源码可以看出createApplicationContext()方法的执行逻辑:根据webApplicationType决定创建哪种contextClass。webApplicationType变量赋值的过程在前文中已经介绍过。因为该类型为WebApplicationType.SERVLET类型,所以会通过反射装载对应的字节码DEFAULF_SERVLET_WEB_CONTEXT_CLASS创建。创建的容器类型为AnnotationConfigServletWebServerApplicationContext,在后续步骤中的操作都会基于该容器。
(11)准备ApplicationContext实例,prepareContext()方法的源码如下所示:
data:image/s3,"s3://crabby-images/09a20/09a209852eca26fae14e5c539e9b8145854651f6" alt=""
data:image/s3,"s3://crabby-images/623f0/623f0a3d73d562376db29fec8ba7d8dcb3a23b88" alt=""
在创建对应的Spring容器后,程序会进行初始化、加载主启动类等预处理工作。至此,主启动类加载完成,容器准备好。
(12)刷新容器,refreshContext()方法的源码如下所示:
data:image/s3,"s3://crabby-images/a427e/a427e198830f77c01bb963248e434601a3c9446a" alt=""
程序首先注册一个Hook函数,然后调用refresh()方法,经过层层调用,程序执行ServletWebServerApplicationContext类中的refresh()方法,源码如下所示:
data:image/s3,"s3://crabby-images/b1ca1/b1ca114a3c7f39b8b356feffc89bc70d4efda7ee" alt=""
data:image/s3,"s3://crabby-images/d369b/d369b553385d521d29c291f204d34823557fb587" alt=""
ServletWebServerApplicationContext会调用父类AbstractApplicationContext的refresh()方法,因此最终执行的refresh()方法源码如下所示:
data:image/s3,"s3://crabby-images/8e7ea/8e7ea13d14a0f317d4989e64ae29001945f0c250" alt=""
data:image/s3,"s3://crabby-images/919a0/919a017b378990d59117c557895e67d832b179f1" alt=""
该方法是Spring Bean加载的核心,用于刷新整个Spring上下文信息,定义整个Spring上下文加载的流程。其包括实例的初始化和属性设置、自动配置类的加载和执行、内置Tomcat服务器的启动等步骤。在后续章节中笔者也会结合源码对这些过程进行介绍。
(13)调用afterRefresh()方法,执行Spring容器初始化的后置逻辑,默认实现是一个空的方法:
data:image/s3,"s3://crabby-images/d11b4/d11b48cc2baa57b0e0c5b97549f665df22b1a8e2" alt=""
(14)代码执行时间的监控停止,即知道了启动应用所花费的时间。
(15)发布容器启动事件。
(16)在ApplicationContext完成启动后,程序会对ApplicationRunner和CommandLineRunner进行回调处理,查找当前ApplicationContex中是否注册有CommandLineRunner,如果有,则遍历执行它们。
另外,在SpringApplication启动过程中,如果出现问题会由异常处理器接管,并对异常进行统一处理,源码如下所示:
data:image/s3,"s3://crabby-images/39bbf/39bbf0f98319b9fac66f691af6f9d4eb04d2bdf9" alt=""
data:image/s3,"s3://crabby-images/5fe32/5fe324eec4f8039bd1ea3c418d205e4dc36702a8" alt=""
本章讲解的源码都来自Spring Boot2.3.7.RELEASE版本,它与其他版本的代码可能有些不同。读者想更好地理解Spring Boot及其启动过程的原理,可以参考本章给出的提示并自行通过debug模式进行调试。理论结合实践才能更好地理解Spring Boot在启动过程中的操作。
通过源码解读和启动流程的介绍,相信读者对于Spring Boot框架有了进一步的认识。Spring Boot的核心依然是Spring。它只是在Spring框架的基础之上,针对Spring应用启动流程进行了规范和封装。Spring的核心启动方法是refresh(),Spring Boot在启动时依然会调用该核心方法。在平时的Spring项目开发中,这些组件通常是通过XML配置文件进行定义和装载的,而Spring Boot将该过程简化并通过自动配置的方式实现该过程,减少了开发人员需要做的配置工作量。它更像是基于Spring框架的一个增强版的应用启动器。