参考:
- https://blog.csdn.net/weixin_42228950/article/details/92386375
- https://www.toutiao.com/i6815381538927018500
什么是Spring的循环依赖?
循环依赖 一定是默认的单例Bean的 属性互相引用的场景。
如在创建TestA类时,构造器需要TestB类,那将去创建TestB,在创建TestB类时又发现需要TestC类,则又去创建TestC,最终在创建TestC时发现又需要TestA,从而形成一个环,就是循环依赖。
比如几个Bean之间的互相引用:
Spring解决 对3种循环依赖都是如何解决的?
1、setter注入方式且注入的对象属性是单例的(我们一般都是使用这种方式注入依赖的)
用setter注入方式注入依赖的对象属性 就不会报错。
为什么用setter注入方式 注入依赖的对象属性 就不会报错了呢 ?
原理是因为:
Spring先是用无参构造实例化 Bean对象,将这些 无参实例化结束的对象 缓存到一个 Map
中,并且Spring提供了获取这个 无参实例化对象引用 的方法。
然后 当Spring 要为实例化的Bean注入对象属性时,就会去 Map中取出存在里面的单例StudentB对象,以此类推,不会出来循环的问题喽。
原理实质: 这里提供了一个三级缓存。三级缓存就是3个Map。第三个Map就是用来存放 最终 实例化完对象属性的Bean的。
2、构造器参数注入对象属性
表示通过构造器注入对象属性 造成的循环依赖,是无法解决的,只能抛出BeanCurrentlyIn CreationException
异常表示循环依赖。
抛异常的原理:
Spring容器会将每一个正在创建的Bean 标识符放在一个“当前创建Bean池”中,Bean标识符在创建过程中将一直保持
在这个池中,因此如果在创建Bean过程中发现自己已经在“当前创建Bean池”里时将抛出
BeanCurrentlyInCreationException异常表示循环依赖;而对于创建完毕的Bean将从“当前创建Bean池”中清除掉。
3、setter注入方式,注入多例的Bean属性
对于"prototype"作用域bean,Spring容器不进行缓存"prototype"作用域的bean 到Map
缓存中,只能抛出BeanCurrentlyIn CreationException
异常表示循环依赖。