所谓循环依赖就是多个Bean之间依赖关系形成一个闭环,例如A->B->C->...->A 这种情况,当然,最简单的循环依赖就是2个Bean之间互相依赖:A->B(A依赖B), B->A(B依赖A) 。在Spring中,如果A->B,那么在创建A的过程中会去创建B,在创建B(或B的依赖)的过程中又发现B->A,这个时候就出现了循环依赖的现象。
这两个条件满足的情况下是没问题的。但是如果是通过构造器依赖,或者不是单例模式的情况下循环依赖就会抛出异常BeanCurrentlyInCreationException。下面从代码层面上解析一下为什么。
为什么最先介绍Prototype的循环依赖呢,因为可以顺便介绍在Spring中创建Bean的流程核心流程:在AbstractFoctory的doGetBean的方法。这个方法很长,这里只写出核心逻辑,并在注解上注明了个人理解:
protected T doGetBean(
final String name, final Class requiredType, final Object[] args, boolean typeCheckOnly)
throws BeansException {
final String beanName = transformedBeanName(name);
Object bean;
//尝试获取单例对象,因为spring大部分的bean都是单例的,所以这里先尝试能否获取。
registered singletons.
Object sharedInstance = getSingleton(beanName);
//单例存在的情况下,那么beanName返回的肯定是单例类,但是这里还需要判断是不是FactoryBean
if (sharedInstance != null && args == null) {
...
//FactoryBean应该返回getObject()对象
bean = getObjectForBeanInstance(sharedInstance, name, beanName, null);
}
else {
//走到这里,有可能beanName是单例模式,但之前并没有实例化,或者是Prototype类型。
//首先判断不是循环依赖,这里的循环依赖指的是Prototype类型
if (isPrototypeCurrentlyInCreation(beanName)) {
throw new BeanCurrentlyInCreationException(beanName);
}
try {
final RootBeanDefinition mbd = getMergedLocalBeanDefinition(beanName);
// 如果是单例,则创建单例模式
if (mbd.isSingleton()) {
// !!!这里是解决单例循环依赖的关键,后面再分析
sharedInstance = getSingleton(beanName, new ObjectFactory