【SpringBoot】IOC如何解决Bean的循环依赖

简介: 面试官: 作为一个Java练习两年半的练习生,IOC如何解决Bean的循环依赖这种老八股已经形成了一套丝滑小连招。

SpringBoot相关文章


SpringBean的生命周期

参照上篇文章【SpringBean的生命周期】,Spring的一些常考问题其实是环环相扣的,IOC容器解决循环依赖也是依靠了一些生命周期的东西。这里可以大体回顾一下。

  • bean生命周期的三个过程
  1. beanDefinition
  2. 实例化:执行了bean的构造方法,bean中依赖的对象还未赋值
  3. 设置属性:给bean中依赖的对象赋值,若被依赖的对象尚未初始化,则先进行该对象的生命周期(递归)
  4. 初始化:执行bean的初始化方法,回调方法等


就算没研究过如何解决循环依赖那你也肯定听说过一个叫做三级缓存的东西,如果真的没听说过,那你现在肯听听说了。。。


三级缓存


三级缓存核心是通过将bean的实例化状态(执行完bean的构造方法)提前暴露存入Map中,如果遇到互相依赖的情况通过在Map中取出实例化尚不完整的bean完成当前实例化。


类似递归的思想,挨个排查依赖的对象是否存在于缓存中,但仅仅适用于单例模式,多例模式的bean不使用缓存,循环依赖无法解决。

/** 一级缓存,缓存bean的地方,bean都是完整可用的 */privatefinalMap<String, Object>singletonObjects=newConcurrentHashMap<>(256);
/** 二级缓存,缓存实例化后不完整的bean*/privatefinalMap<String, Object>earlySingletonObjects=newHashMap<>(16);
/** 三级缓存,缓存key为beanName, value为ObjectFactory处理aop*/privatefinalMap<String, ObjectFactory<?>>singletonFactories=newHashMap<>(16);

举个例子,假如A类中引入了B类,B类又引入了A类。那么bean创建时的步骤:

  1. 实例化A的时候,调用A的构造方法,创建出A对象存放到三级缓存中,虽然属性都没有初始化,但是已经注册在IOC容器中。
  2. 对A对象设置属性时,发现A对象依赖了B对象,那么就去创建B对象,按照流程再走一遍实例化,同样会把B实例化不完整对象放入三级缓存中。当B设置属性的时候,发现依赖了A类,这时候再去创建A对象,发现在三级缓存(singletonFactories)能找到A的实例化不完整对象。
  3. 此时会通过A的ObjectFactory获取A,并把A从三级缓存移到二级缓存。然后就可以把B的A属性赋值了,这个时候B就初始化完成了,初始化完成后就会把B从三级缓存移到一级缓存。
  4. 完成B实例化后,回到A调用的设置属性流程中。返回的就是B对象了,对A的B属性进行赋值就可以了。


为什么需要三级缓存

按照上述的描述,好像两个缓存就可以满足循环依赖问题,一个存放半状态bean,一个存放完整bean。不过前提是这个bean没有被AOP进行代理。


三级缓存本质上是为了存放bean的代理对象保证代理对象的单例属性,还是回到刚才的A类和B类相互依赖的场景下,在第三步通过A的ObjectFactory接口获取A对象本质上是通过调用getEarlyBeanReference()方法,这个方法会返回一个A类的AOP对象回去。


问题出在每次调用getEarlyBeanReference方法,每次都会返回一个新的AOP对象,假如没有二级缓存,C类也依赖A类,调用getEarlyBeanReference()方法则注入了一个新的A类的AOP对象,并且与B类注入的A类对象是两个不同的对象,完全违背单例模式。


相关文章
|
8天前
|
XML Java 开发者
Spring Boot中的bean注入方式和原理
Spring Boot中的bean注入方式和原理
146 0
|
8天前
|
存储 NoSQL Java
Spring Boot统计一个Bean中方法的调用次数
Spring Boot统计一个Bean中方法的调用次数
45 1
|
8天前
|
NoSQL Java Redis
SpringBoot-引入Redis依赖
本文介绍如何在IDEA里将SpringBoot整合Redis。
122 0
|
8天前
|
XML Java 数据库
【SpringBoot:详解Bean装配】
【SpringBoot:详解Bean装配】
11 3
|
8天前
|
Java 数据库连接 容器
SpringBoot之IOC&DI的详细解析
SpringBoot之IOC&DI的详细解析
11 0
SpringBoot之IOC&DI的详细解析
|
8天前
|
存储 Java 程序员
SpringBoot之分层解耦以及 IOC&DI的详细解析
SpringBoot之分层解耦以及 IOC&DI的详细解析
15 0
|
8天前
|
Java Spring 容器
SpringBoot 使用Quartz执行定时任务对象时无法注入Bean问题
SpringBoot 使用Quartz执行定时任务对象时无法注入Bean问题
22 1
|
8天前
|
Java Spring 容器
SpringBoot中bean的生命周期
Spring Boot的Bean生命周期涉及实例化、属性注入、初始化和销毁。在实例化后,Spring通过构造函数或Setter注入属性,然后调用初始化方法(@PostConstruct、InitializingBean接口)。Bean在应用中使用后,当容器关闭时,会调用销毁方法(@PreDestroy、DisposableBean接口)。依赖注入、配置管理、组件扩展和切面编程是其常见应用场景。示例代码展示了如何通过实现BeanNameAware、BeanFactoryAware等接口以及使用@PostConstruct注解来控制Bean的初始化。
46 2
SpringBoot中bean的生命周期
|
8天前
|
存储 前端开发 Java
springboot中的第二个IOC容器BootstrapContext
springboot中的第二个IOC容器BootstrapContext
springboot中的第二个IOC容器BootstrapContext
|
8天前
|
XML Java 数据格式
【springboot原理篇】Bean的加载方式,面试必看
【springboot原理篇】Bean的加载方式,面试必看