springboot-容器及启动过程

SpringBoot会启动两个容器,即父子容器
父容器(Spring容器):AnnotationConfigWebApplicationContex
子容器(SpringMVC容器):AnnotationConfigEmbeddedWebApplicationContex

思考几个问题:
问题1:为什么要启动父子两个容器?
在Spring整体框架的核心概念中,容器是核心思想,而在一个项目中,容器不一定只有一个,Spring中可以包括多个容器,使用上下层关系来管理容器可以解决公共属性抽取的问题,即将公共的属性放到父容器中,对其各个子容器均可见,实现了复用功能,注意子容器对父容器是不可见的。
问题2:在什么时候启动父容器?
准备环境变量,即执行prepareEnvironment方法时,发布ApplicationEnvironmentPreparedEvent事件,BootstrapApplicationListener监听器收到该事件,会初始化父容器
问题3:父容器和子容器的区别?
1.子容器中会创建植入的tomcat容器类,用来管理tomcat服务,父容器没有
2.父容器管理的bean集合是spring框架内置的一些bean,子容器管理的bean集合是我们业务定义的bean以及spring框架内置的bean
问题4:怎么保证父容器启动过程中,执行prepareEnvironment方法时,BootstrapApplicationListener监听器不再重复执行?
父容器初始化完成后,会在webEnvironment属性中加入bootstrap属性标识,这样就可以根据该标识来判断是否需要执行了
问题5:容器实际通过什么来管理bean
容器通过beanFactory来管理对象,父子容器都会各持有一个DefaultListableBeanFactory,即父子容器都会管理各自的bean。

子容器启动流程:
1.执行初始化:初始化webEnvironment以及listeners
2.发布容器启动事件
3.在执行prepareEnvironment步骤时,发布ApplicationEnvironmentPreparedEvent事件,BootstrapApplicationListener监听器收到该事件,会再次调用SpringApplication
  初始化和run方法(初始化父容器的步骤类似子容器)
4.创建context,通过判断webEnvironment标识,这时为true,创建AnnotationConfigEmbeddedWebApplicationContext实例
5.执行refreshContext,比较核心的方法,详细说一下:
  1.获取bean工厂,即创建DefaultListableBeanFactory实例
  2.为bean工厂赋一些基础属性,主要有类加载器
  3.为bean工厂注册WebApplicationContextServletContextAwareProcessor,一个BeanPostProcessor,主要用于获取ServletContext
  4.执行beanFactoryPostProcessor,递归处理,主要用于扫描@CompontScan、@Import、@ImportSource等注解的类,首先注册bean元数据到工厂,然后可以修改bean工厂
  5.实例化BeanPostProcessor类并注册到bean工厂单例列表中
  6.实例化消息通知类并注册到bean工厂单例列表中
  7.调用onRefresh方法,初始化一个植入的tomcat容器
  8.注册监听器到消息通知器中,区分为两类:一、从系统配置中加载的监听器(已实例化) 二、已注册但还未实例化的监听器
  9.实例化注册表中的所有bean,创建bean的过程中会执行bean的初始化操作,会执行BeanPostProcessor、afterPropertiesSet、初始化方法等等
  10.调用finishRefresh,启动所有bean,发布ContextRefreshed事件
6.执行afterRefresh,调用实现了ApplicationRunner/CommandLineRunner接口的bean run方法
7.发布ApplicationReadyEvent事件

你可能感兴趣的:(springboot)