Sring容器启动流程

  1. 容器准备阶段
    1. 获取一个默认的DefaultListableBeanFactory 的BeanFactory.
    2. 扫描注册一些内置的BeanPostProcessor,这些后置处理器是对容器的功能提供支持,比如
      ApplicationContextAwareProcessor,是为了确保其余处理器在执行的时候如果要获取上下文确保能获取到。
    3. 扫描注册BeanFactoryPostProcessor扫描容器的后置处理器并且注册。BeanDefinitionRegistryPostProcessor就是一个容器后置处理器,能够修改Bean的定义信息
    4. 扫描和注册Bean的定义信息BeanDefinition
  2. 执行容器后置处理器BeanFactoryPostProcessor
    1. 执行普通的容器后置处理器
    2. 执行BeanDefinitionRegistryPostProcessor,修改bean的定义信息
  3. 实例化BeanBean的后置处理器发挥作用
    1. 实例化之前会调用`InstantiationAwareBeanPostProcessor` 类型的处理器的 `postProcessBeforeInstantiation()` 方法实现代理的时机

      不过基本不会在这一步进行对象的代理,这个时候对象的属性还没赋值,可能还有自定义操作没有完成,得到的是半成品。

    2. 实例化Bean

    3. 实例化之前会调用InstantiationAwareBeanPostProcessor 类型的处理器的postProcessAfterInstantiation() 方法;

      根据返回结果决定是否继续属性赋值。

    4. 在属性注入之前,执行InstantiationAwareBeanPostProcessor 接口中的方法postProcessPropertyValues() ,能够对属性进行操作

  4. 属性赋值
  5. 执行部分特殊的BeanPostProcessor,例如ApplicationContextAware,会执行相应的方法。
  6. 初始化Bean
    1. 初始化之前执行所有注册的 BeanPostProcessorpostProcessBeforeInitialization() 方法。大部分的后置处理器在这里执行,包括自定义的,内置的处理器在各个阶段都有执行。

    2. 初始化Bean,执行初始化方法,如init-method 或通过 @PostConstruct 注解标注的方法。

    3. 初始化之后执行所有注册的 BeanPostProcessorpostProcessAfterInitialization() 方法。

      代理对象的创建一般是在这个时候创建的,这时候目标对象已经设置完成,可以代理
  7. 注册销毁方法
  8. 发布容器刷新完成事件

循环依赖以及缓存

循环依赖

原因

例如,类 A 依赖类 B,同时类 B 又依赖类 A。在对象创建和依赖注入的过程中,如果不加以处理,就会陷入死循环。

解决办法

spring引入了多级缓存机制,在实例化结束后就把对象放入到二级缓存,可以拿到这个不完全的对象来使用。

存在问题

二级缓存中存放的是真正的目标对象,因为代理对象要在初始化之后才会创建,所以这时候只有目标对象,这个时候如果被代理的对象被注入的话,拿到的就是目标对象,代理功能就无法使用了。

代理对象解决办法

使用三级缓存,在实例化对象之后会把一个对象工厂放入到三级缓存。三级缓存不仅仅是为了延迟实例化,它的另一个重要作用是处理 代理问题 和确保 完整初始化。工厂方法得到的Bean确保了完全初始化并且代理逻辑能够实现。然后将代理对象放到二级缓存供别的对象注入。