Spring如何解决循环依赖


Spring容器循环依赖包括构造器循环依赖和setter循环依赖,那Spring容器如何解决循环依赖呢?首先让我们来定义循环引用类:

public class TestA {  
 
    private TestB testB;  
 
    public void a() {  
        testB.b();  
    }  
 
    public TestB getTestB() {  
        return testB;  
    }  
 
    public void setTestB(TestB testB) {  
        this.testB = testB;  
    }  
}  
 
public class TestB {  
    private TestC testC;  
 
    public void b() {  
        testC.c();  
    }  
 
    public TestC getTestC() {  
        return testC;  
    }  
 
    public void setTestC(TestC testC) {  
        this.testC = testC;  
    }  
}  
 
 
public class TestC {  
    private TestA testA;  
 
    public void c() {  
        testA.a();  
    }  
 
    public TestA getTestA() {  
        return testA;  
    }  
 
    public void setTestA(TestA testA) {  
        this.testA = testA;  
    }  
}  


在Spring中将循环依赖的处理分成了3种情况。

1.构造器循环依赖

表示通过构造器注入构成的循环依赖,此依赖是无法解决的,只能抛出BeanCurrentlyIn CreationException异常表示循环依赖。

如在创建TestA类时,构造器需要TestB类,那将去创建TestB,在创建TestB类时又发现需要TestC类,则又去创建TestC,最终在创建TestC时发现又需要TestA,从而形成一个环,没办法创建。

Spring容器将每一个正在创建的bean标识符放在一个"当前创建bean池"中,bean标识符在创建过程中将一直保持在这个池中,因此如果在创建bean过程中发现自己已经在"当前创建bean池"里时,将抛出BeanCurrentlyInCreationException异常表示循环依赖;而对于创建完毕的bean将从"当前创建bean池"中清除掉。

我们通过一个直观的测试用例来进行分析。

(1)创建配置文件。

    <bean id="testA" class="com.bean.TestA">     
        <constructor-arg index="0" ref="testB"/>     
    </bean>     
    <bean id="testB" class="com.bean.TestB">     
        <constructor-arg index="0" ref="testC"/>     
    </bean>     
    <bean id="testC" class="com.bean.TestC">     
        <constructor-arg index="0" ref="testA"/>     
    </bean>     


(2)创建测试用例。

    @Test(expected = BeanCurrentlyInCreationException.class)     
    public void testCircleByConstructor() throws Throwable {     
        try {     
                 new ClassPathXmlApplicationContext("test.xml");     
        } catch (Exception e) {     
          //因为要在创建testC时抛出;     
          Throwable ee1 = e.getCause().getCause().getCause();     
          throw e1;     
        }     
    }   

针对以上代码的分析如下。

Spring容器创建"testA"bean,首先去"当前创建bean池"查找是否当前bean正在创建,如果没发现,则继续准备其需要的构造器参数"testB",并将"testA"标识符放到"当前创建bean池"。

Spring容器创建"testB"bean,首先去"当前创建bean池"查找是否当前bean正在创建,如果没发现,则继续准备其需要的构造器参数"testC",并将"testB"标识符放到"当前创建bean池"。

Spring容器创建"testC"bean,首先去"当前创建bean池"查找是否当前bean正在创建,如果没发现,则继续准备其需要的构造器参数"testA",并将"testC"标识符放到"当前创建Bean池"。

到此为止Spring容器要去创建"testA"bean,发现该bean标识符在"当前创建bean池"中,因为表示循环依赖,抛出BeanCurrentlyInCreationException。

2.setter循环依赖

表示通过setter注入方式构成的循环依赖。对于setter注入造成的依赖是通过Spring容器提前暴露刚完成构造器注入但未完成其他步骤(如setter注入)的bean来完成的,而且只能解决单例作用域的bean循环依赖。通过提前暴露一个单例工厂方法,从而使其他bean能引用到该bean,如下代码所示:

addSingletonFactory(beanName, new ObjectFactory() {     
    public Object getObject() throws BeansException {     
        return getEarlyBeanReference(beanName, mbd, bean);     
    }     
});   

具体步骤如下。

(1)Spring容器创建单例"testA"bean,首先根据无参构造器创建bean,并暴露一个"ObjectFactory"用于返回一个提前暴露一个创建中的bean,并将"testA"标识符放到"当前创建bean池",然后进行setter注入"testB"。

(2)Spring容器创建单例"testB"bean,首先根据无参构造器创建bean,并暴露一个"ObjectFactory"用于返回一个提前暴露一个创建中的bean,并将"testB"标识符放到"当前创建bean池",然后进行setter注入"circle"。

(3)Spring容器创建单例"testC"bean,首先根据无参构造器创建bean,并暴露一个"ObjectFactory"用于返回一个提前暴露一个创建中的bean,并将"testC"标识符放到"当前创建bean池",然后进行setter注入"testA"。进行注入"testA"时由于提前暴露了"ObjectFactory"工厂,从而使用它返回提前暴露一个创建中的bean。

(4)最后在依赖注入"testB"和"testA",完成setter注入。

3.prototype范围的依赖处理

对于"prototype"作用域bean,Spring容器无法完成依赖注入,因为Spring容器不进行缓存"prototype"作用域的bean,因此无法提前暴露一个创建中的bean。示例如下:

(1)创建配置文件。

<bean id="testA" class="com.bean.CircleA" scope="prototype">     
      <property name="testB" ref="testB"/>     
 </bean>     
 <bean id="testB" class="com.bean.CircleB" scope="prototype">     
     <property name="testC" ref="testC"/>     
 </bean>     
 <bean id="testC" class="com.bean.CircleC" scope="prototype">     
    <property name="testA" ref="testA"/>     
 </bean>   

(2)创建测试用例。
    @Test(expected = BeanCurrentlyInCreationException.class)     
    public void testCircleBySetterAndPrototype () throws Throwable {     
        try {     
            ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext(     
    "testPrototype.xml");     
            System.out.println(ctx.getBean("testA"));     
        } catch (Exception e) {     
            Throwable ee1 = e.getCause().getCause().getCause();     
            throw e1;     
        }     
    }    

针对以上代码的分析如下。

Spring容器创建"testA"bean,首先去"当前创建bean池"查找是否当前bean正在创建,如果没发现,则继续准备其需要的构造器参数"testB",并将"testA"标识符放到"当前创建bean池"。

Spring容器创建"testB"bean,首先去"当前创建bean池"查找是否当前bean正在创建,如果没发现,则继续准备其需要的构造器参数"testC",并将"testB"标识符放到"当前创建bean池"。

Spring容器创建"testC"bean,首先去"当前创建bean池"查找是否当前bean正在创建,如果没发现,则继续准备其需要的构造器参数"testA",并将"testC"标识符放到"当前创建Bean池"。

到此为止Spring容器要去创建"testA"bean,发现该bean标识符在"当前创建bean池"中,因为表示循环依赖,抛出BeanCurrentlyInCreationException。

2.setter循环依赖

表示通过setter注入方式构成的循环依赖。对于setter注入造成的依赖是通过Spring容器提前暴露刚完成构造器注入但未完成其他步骤(如setter注入)的bean来完成的,而且只能解决单例作用域的bean循环依赖。通过提前暴露一个单例工厂方法,从而使其他bean能引用到该bean,如下代码所示:

 
 
  1. addSingletonFactory(beanName, new ObjectFactory() {     
  2.     public Object getObject() throws BeansException {     
  3.         return getEarlyBeanReference(beanName, mbd, bean);     
  4.     }     
  5. });    

具体步骤如下。

(1)Spring容器创建单例"testA"bean,首先根据无参构造器创建bean,并暴露一个"ObjectFactory"用于返回一个提前暴露一个创建中的bean,并将"testA"标识符放到"当前创建bean池",然后进行setter注入"testB"。

(2)Spring容器创建单例"testB"bean,首先根据无参构造器创建bean,并暴露一个"ObjectFactory"用于返回一个提前暴露一个创建中的bean,并将"testB"标识符放到"当前创建bean池",然后进行setter注入"circle"。

(3)Spring容器创建单例"testC"bean,首先根据无参构造器创建bean,并暴露一个"ObjectFactory"用于返回一个提前暴露一个创建中的bean,并将"testC"标识符放到"当前创建bean池",然后进行setter注入"testA"。进行注入"testA"时由于提前暴露了"ObjectFactory"工厂,从而使用它返回提前暴露一个创建中的bean。

(4)最后在依赖注入"testB"和"testA",完成setter注入。

3.prototype范围的依赖处理

对于"prototype"作用域bean,Spring容器无法完成依赖注入,因为Spring容器不进行缓存"prototype"作用域的bean,因此无法提前暴露一个创建中的bean。示例如下:

(1)创建配置文件。

 
 
  1. <bean id="testA" class="com.bean.CircleA" scope="prototype">     
  2.       <property name="testB" ref="testB"/>     
  3.  </bean>     
  4.  <bean id="testB" class="com.bean.CircleB" scope="prototype">     
  5.      <property name="testC" ref="testC"/>     
  6.  </bean>     
  7.  <bean id="testC" class="com.bean.CircleC" scope="prototype">     
  8.     <property name="testA" ref="testA"/>     
  9.  </bean>    

(2)创建测试用例。

 
 
  1. @Test(expected = BeanCurrentlyInCreationException.class)     
  2. public void testCircleBySetterAndPrototype () throws Throwable {     
  3.     try {     
  4.         ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext(     
  5. "testPrototype.xml");     
  6.         System.out.println(ctx.getBean("testA"));     
  7.     } catch (Exception e) {     
  8.         Throwable ee1 = e.getCause().getCause().getCause();     
  9.         throw e1;     
  10.     }     
  11. }    

对于"singleton"作用域bean,可以通过"setAllowCircularReferences(false);"来禁用循环引用。



  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值