浅谈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的代理对象的,所以会导致两者不一致。

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

相关推荐
heRs BART2 小时前
Redis简介、常用命令及优化
数据库·redis·缓存
@ chen2 小时前
IDEA初始化配置
java·ide·intellij-idea
蒸汽求职2 小时前
破局“无效互面”:跨国大厂视角的工业级 Mock Interview 价值解析
缓存·面试·职场和发展·金融·notion
wellc2 小时前
SpringBoot集成Flowable
java·spring boot·后端
Irissgwe3 小时前
redis之常见数据类型
数据库·redis·缓存
Hui Baby3 小时前
springAi+MCP三种
java
hsjcjh3 小时前
【MySQL】C# 连接MySQL
java
敖正炀3 小时前
LinkedBlockingDeque详解
java
wangyadong3173 小时前
datagrip 链接mysql 报错
java