Spring为何需要三级缓存解决循环依赖,而不是二级缓存?
今天给大家分享一道大厂面试真题,Spring为何需要三级缓存解决循环依赖,而不是二级缓存?我一共分为五个部分来给大家介绍:
1、什么是循环依赖?
循环依赖就是指循环引用,是两个或多个Bean相互之间的持有对方的引用。在代码中,如果将两个或多个Bean互相之间持有对方的引用,因为Spring中加入了依赖注入机制,也就是自动给属性赋值。Spring给属性赋值时,将会导致死循环。那么,哪些情况会出现循环依赖呢?
2、哪些情况会出现循环依赖?
循环依赖有三种形态:
1、相互依赖,也就是A 依赖 B,B 又依赖 A,它们之间形成了循环依赖。
2、三者间依赖,也就是A 依赖 B,B 依赖 C,C 又依赖 A,形成了循环依赖。
3、自我依赖,也是A依赖A形成了循环依赖自己依赖自己。
3、Spring如何解决循环依赖问题?
Spring中设计了三级缓存来解决循环依赖问题,当我们去调用getBean()方法的时候,Spring会先从一级缓存中去找到目标Bean,如果发现一级缓存中没有 便会去二级缓存中去找,而如果一、二级缓存中都没有找到,意味着该目标Bean还没有实例化。于是,Spring容器会实例化目标Bean(PS:刚初始化的Bean称为早期Bean),然后,将目标Bean放入到二级缓存中,同时,加上标记是否存在循环依赖。如果不存在循环依赖便会将目标Bean存入到二级缓存,否则,便会标记该Bean存在循环依赖,然后将等待下一次轮询赋值,也就是解析@Autowired注解。等@Autowired注解赋值完成后(PS:完成赋值的Bean称为成熟Bean),会将目标Bean存入到一级缓存。
总结一下,Spring一级缓存中存放所有的成熟Bean,二级缓存中存放所有的早期Bean,先取一级缓存,再去二级缓存。
4、为何需要三级缓存,而不是二级缓存?
那么,前面有提到三级缓存,三级缓存的作用是啥呢?来看这样一张图,三级缓存是用来存储代理Bean,当调用getBean()方法时,发现目标Bean需要通过代理工厂来创建,此时会将创建好的实例保存到三级缓存,最终也会将赋值好的Bean同步到一级缓存中。大家可以私信我或者在评论区留言获取高清大图。
5、Spring中哪些情况下,不能解决循环依赖问题?
1.多例Bean通过setter注入的情况,不能解决循环依赖问题
2.构造器注入的Bean的情况,不能解决循环依赖问题
3.单例的代理Bean通过Setter注入的情况,不能解决循环依赖问题
4.设置了@DependsOn的Bean的情况,不能解决循环依赖问题
我是被编程耽误的文艺Tom,如果大家还有其他疑问,请在评论区留言。如果本次面试对你有帮助,请动动手指一键三连分享给更多的人。关注我,面试不再难!
本文为“Tom弹架构”原创,转载请注明出处。技术在于分享,我分享我快乐!\ 如果本文对您有帮助,欢迎关注和点赞;如果您有任何建议也可留言评论或私信,您的支持是我坚持创作的动力。
关注微信公众号『 Tom弹架构 』可获取更多技术干货!往期视频已经整理成文档形式,需要的小伙伴点个关注,搜索下方名片!我是被编程耽误的文艺Tom,
- 使用桥接模式设计复杂的消息系统
- 为什么MySQL索引结构采用B 树?
- 为什么Netty线程池默认大小为CPU核数的2倍
- 谈谈你对深克隆和浅克隆的理解
- 什么是代理,为什么要用动态代理?
- 什么是零拷贝,Netty是如何实现的?
- 3分钟轻松理解单线程下的HashMap工作原理
- 被面试官问烂的Spring AOP原理,你是怎么答的?
- Spring为何需要三级缓存解决循环依赖,而不是二级缓存?
- 为什么Spring中每个Bean都要定义作用域?
- 谈谈你对Spring Bean的理解
- 趣谈装饰器模式,让你一辈子不会忘
- 掌握这些招数,你也能写出HR眼中的高分简历
- MongoDB高级应用之数据转存与恢复(5)
- 图解MongoDB集群部署原理(3)
- 爆肝30天,肝出来史上最透彻Spring原理和27道高频面试题总结
- Spring核心原理之IoC容器初体验(2)
- Spring核心原理分析之MVC九大组件(1)
- 30个类手写Spring核心原理之动态数据源切换(8)
- 【紧急】Log4j又发新版2.17.0,只有彻底搞懂漏洞原因,才能以不变应万变,小白也能看懂