循环依赖有三种情况:构造器循环依赖,setter循环依赖和prototype范围的循环依赖
1、构造器循环依赖
当使用构造器注入时,也就是使用<constructor-arg>标签进行注入时,A引用了B,B引用了C,C又引用了A时,这种循环依赖是没办法解决的,因为在A构造器中引用B类时,B类很可能还没创建完,而B需要用构造器构造并引用C,C又想去引用A,但A却因为B,C没有创建完而迟迟无法结束,这样将会一直创建下去直到内存不足。
所以spring在这种情况下只能靠抛出异常来解决这个问题,容器会将每一个正在创建的bean标识符放在一个”当前正在创建的bean池“中,当创建一个bean实例时,会先去检查是否在该池中已经存在一个正在创建中的该bean了,此时就会抛出BeanCurrentlyInCreationException异常来表示循环依赖。
2、setter循环依赖
针对setter注入的情况,因为可以先直接使用无参构造来构造对象,然后再调用set方法来注入,所以setter注入导致的循环依赖,是可以解决的。
spring针对这种情况,会在对象还未创建完之前提前暴露一个单例工厂方法,从而使其他的bean能够引用到该bean
3、prototype范围的循环依赖
protorype作用域的bean,spring是不支持缓存protorype类型的bean的,所以无法提前暴露一个创建中的bean,所以spring对于这种情况依然和1一样,抛出BeanCurrentlyInCreationException异常来表示循环依赖。