目录
背景
22年11月底,应聘杭州某外包公司的驻场开发。
题目的答案,是根据我的理解和整理编写的。可能有不对的地方,欢迎大家指正。
这次是电话面,我项目介绍比较充分,也可能是面试官赶时间,除了项目细节,面试官只问了一道面试题。
题目
Spring怎么解决循环依赖?
Spring 是利用缓存机制来解决循环依赖。
什么是循环依赖
如果在代码中,将两个或多个 Bean 互相之间持有对方的引用就会发生循环依赖。
循环的依赖将会导致注入死循环。
这是 Spring 发生循环依赖的原因。
循环依赖有三种形态:
第一种:互相依赖
A 依赖 B,B 又依赖 A,它们之间形成了循环依赖。
第二种:三者间依赖
A 依赖 B,B 依赖 C,C 又依赖 A,形成了循环依赖。
第三种:自我依赖
A 依赖 A 形成了循环依赖。
三级缓存
Spring 设计了三级缓存来解决循环依赖问题。
当调用 getBean()方法的时候,Spring 会先从一级缓存中去找到目标 Bean。
如果发现一级缓存中没有便会去二级缓存中去找,而如果一、二级缓存中都没有找到,意味着该目标 Bean还没有实例化。
于是,Spring 容器会实例化目标 Bean(PS:刚初始化的 Bean称为早期 Bean) 。
然后,将目标 Bean 放入到二级缓存中,同时,加上标记是否存在循环依赖。
如果不存在循环依赖便会将目标 Bean 存入到二级缓存,否则,便会标记该 Bean 存在循环依赖,然后将等待下一次轮询赋值,也就是解析@Autowired 注解。等@Autowired 注解赋值完成后,会将目标 Bean 存入到一级缓存。Spring 一级缓存中存放所有的成熟 Bean,二级缓存中存放所有的早期 Bean,先取一级缓存,再去二级缓存。
补充:那第三级缓存的作用是什么?
三级缓存是用来存储代理 Bean。
当调用 getBean()方法时,发现目标 Bean 需要通过代理工厂来创建,此时会将创建好的实例保存到三级缓存,最终也会将赋值好的 Bean 同步到一级缓存中。
补充:Spring 中哪些情况下,不能解决循环依赖问题?
有4种情况:
1.多例 Bean 通过 setter 注入
2.单例的代理 Bean 通过 Setter 注入
3.构造器注入的 Bean
4.设置了@DependsOn 的 Bean