Spring如何去解决循环依赖问题的?
1.什么是循环依赖?
循环依赖就是循环引用,就是两个或多个Bean相互之间的持有对方的引用,循环的依赖将会导致陷入死循环。这就是Spring发生循环依赖的问题。
循环依赖有三种形式:
- A依赖B,B又依赖A它们之间形成了互相依赖
- A 依赖B,B依赖C,C又依赖A,形成了三者间依赖
- A 依赖A 形成了自我依赖
2.如何去解决循环依赖?
Spring中设计了三级缓存来解决循环依赖的问题,当我们去调用getBean()方法的时候,Spring会先从一级缓存中去找到目标 Bean,如果发现一级缓存中没有便会去二级缓存中去找,而如果一、二级缓存中都没有找到,意味着该目标 Bean 还没有实例化。于是,Spring 容器会实例化目标 Bean。然后,将目标 Bean 放入二级缓存中,同时,加上标记是否存在循环依赖。如果不存在循环依赖便会将目标 Bean 存入到二级缓存,否则,便会标记该 Bean 存在循环依赖,然后将等待下一次轮询赋值,也就是解析@Autowired 注解。等@Autowired 注解赋值完成后(PS:完成赋值的 Bean 称为成熟 Bean)会将目标 Bean 存入到一级缓存。
Spring 一级缓存中存放所有的成熟 Bean级缓存中存放所有的早期 Bean,先取一级缓存,再去二级缓存。
3.三级缓存的作用是什么?
三级缓存是用来存储代理 Bean,当调用 getBean()方法时,发现目标 Bean 需要通过代理工厂来创建此时会将创建好的实例保存到三级缓存,最终也会将赋值好的 Bean 同步到一级缓存中。]
4.Spring 中哪些情况下,不能解决循环依赖问题?
- 多例 Bean 通过 setter 注入的情况,不能解决循环依赖问题
- 构造器注入的 Bean 的情况,不能解决循环依赖问题
- 单例的代理 Bean 通过 Setter 注入的情况,不能解决循环依赖问题
- 设置了@DependsOn 的 Bean 的情况,不能解决循环依赖问题