浅谈Spring三级缓存

目录

1.什么是循环依赖?

2.三种常见形式

3.图解

4.避免出现循环依赖的方式

5.补充


最近一直在面试,上周遇到这个问题,因为之前其实梳理过这个问题,但是因为时间过去很久,一些细节给忽略掉了,今天正好找到了之前画的示意图,针对Spring的循环依赖问题的解决方案,再尝试自己去理解下,可能不是很周全,如有错的地方,还望指正。

开门见山,就不再赘述什么是循环依赖,简单提一下就是A服务依赖B服务,同时B服务也依赖A服务,当然也有三者之间的闭环依赖,总之就是依赖之间的闭环。

1.什么是循环依赖?

例如:两个或多个 Bean 互相持有对方的引用,形成依赖闭环:

java 复制代码
@Component
public class BeanA {
    @Autowired
    private BeanB beanB;
}

@Component
public class BeanB {
    @Autowired
    private BeanA beanA;
}

2.三种常见形式

注入方式 是否能被 Spring 解决 说明
Setter/Field 注入(单例) ✅ 能解决 最常见场景,也是三级缓存的主战场
构造器注入 ❌ 无法解决 实例化阶段就必须拿到完整对象,无机会提前暴露
多例 Bean ❌ 无法解决 不缓存 Bean,每次 getBean 都新建,会无限递归

3.图解

4.避免出现循环依赖的方式

  • 拆分公共逻辑到新类 C,让 A 和 B 都依赖 C,而不是互相依赖
  • 用依赖倒置原则,把高层模块的依赖抽象出来
  • 把构造器注入改成 @Autowired 字段 /setter 注入
  • 延迟注入:用 @Lazy 注解,注入时不立即初始化 Bean

5.补充

面试的过程中,提到一个为什么不使用二级缓存,为什么是三级缓存,解决了什么问题,当时说实话,这块确实忘了,今天再看才回忆起来,其实是因为要解决的是AOP的代理,因为二级缓存确实是可以解决了循环依赖,但是考虑到如果存在AOP代理的话,只用二级缓存是可能出现原始对象和代理对象不统一的,什么意思呢?注入的是a服务的半成品,此时AOP都是不生效的,但是最后生成了完整的a服务bean的时候是拥有AOP的代理对象的,所以会导致两者不一致。

以上是我个人的见解,欢迎交流留言!

相关推荐
张不才2 小时前
CPU 100% 了怎么办?Java 性能排障的标准化操作
java·后端
shepherd1114 小时前
吞吐量提升 10 倍:高并发大批量数据处理任务的架构演进与性能调优
java·后端·架构
plainGeekDev6 小时前
单例模式 → object 声明
android·java·kotlin
用户298698530147 小时前
Java 实现 Word 文档文本与图片提取的方法
java·后端
SimonKing8 小时前
铁子,IntelliJ IDEA 2026.1.3来了,升不升?
java·后端·程序员
咖啡八杯19 小时前
GoF设计模式——策略模式
java·后端·spring·设计模式
用户128526116021 天前
我把祖传Java项目重构后,接口响应从3s砍到了200ms,只改了这几行代码
java
Linsk1 天前
组件 = 模板 + 业务逻辑
java·前端·vue.js
星沉远浦1 天前
用Gemini高效解决Java代码报错难以定位的问题
java