Spring如何解决循环依赖

来源:互联网 发布:e5a安卓中文编程软件 编辑:程序博客网 时间:2024/05/18 17:42

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

  1.  public class TestA {  
  2.  
  3.     private TestB testB;  
  4.  
  5.     public void a() {  
  6.         testB.b();  
  7.     }  
  8.  
  9.     public TestB getTestB() {  
  10.         return testB;  
  11.     }  
  12.  
  13.     public void setTestB(TestB testB) {  
  14.         this.testB = testB;  
  15.     }  
  16. }  
  17.  
  18. public class TestB {  
  19.     private TestC testC;  
  20.  
  21.     public void b() {  
  22.         testC.c();  
  23.     }  
  24.  
  25.     public TestC getTestC() {  
  26.         return testC;  
  27.     }  
  28.  
  29.     public void setTestC(TestC testC) {  
  30.         this.testC = testC;  
  31.     }  
  32. }  
  33.  
  34.  
  35. public class TestC {  
  36.     private TestA testA;  
  37.  
  38.     public void c() {  
  39.         testA.a();  
  40.     }  
  41.  
  42.     public TestA getTestA() {  
  43.         return testA;  
  44.     }  
  45.  
  46.     public void setTestA(TestA testA) {  
  47.         this.testA = testA;  
  48.     }  
  49. }  

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

1.构造器循环依赖

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

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

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

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

(1)创建配置文件。

  1. <bean id="testA" class="com.bean.TestA">     
  2.     <constructor-arg index="0" ref="testB"/>     
  3. </bean>     
  4. <bean id="testB" class="com.bean.TestB">     
  5.     <constructor-arg index="0" ref="testC"/>     
  6. </bean>     
  7. <bean id="testC" class="com.bean.TestC">     
  8.     <constructor-arg index="0" ref="testA"/>     
  9. </bean>     

(2)创建测试用例。

  1. @Test(expected = BeanCurrentlyInCreationException.class)     
  2. public void testCircleByConstructor() throws Throwable {     
  3.     try {     
  4.              new ClassPathXmlApplicationContext("test.xml");     
  5.     } catch (Exception e) {     
  6.       //因为要在创建testC时抛出;     
  7.       Throwable ee1 = e.getCause().getCause().getCause();     
  8.       throw e1;     
  9.     }     
  10. }   
  11. Spring容器循环依赖包括构造器循环依赖和setter循环依赖,那Spring容器如何解决循环依赖呢?首先让我们来定义循环引用类:

    1.  public class TestA {  
    2.  
    3.     private TestB testB;  
    4.  
    5.     public void a() {  
    6.         testB.b();  
    7.     }  
    8.  
    9.     public TestB getTestB() {  
    10.         return testB;  
    11.     }  
    12.  
    13.     public void setTestB(TestB testB) {  
    14.         this.testB = testB;  
    15.     }  
    16. }  
    17.  
    18. public class TestB {  
    19.     private TestC testC;  
    20.  
    21.     public void b() {  
    22.         testC.c();  
    23.     }  
    24.  
    25.     public TestC getTestC() {  
    26.         return testC;  
    27.     }  
    28.  
    29.     public void setTestC(TestC testC) {  
    30.         this.testC = testC;  
    31.     }  
    32. }  
    33.  
    34.  
    35. public class TestC {  
    36.     private TestA testA;  
    37.  
    38.     public void c() {  
    39.         testA.a();  
    40.     }  
    41.  
    42.     public TestA getTestA() {  
    43.         return testA;  
    44.     }  
    45.  
    46.     public void setTestA(TestA testA) {  
    47.         this.testA = testA;  
    48.     }  
    49. }  

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

    1.构造器循环依赖

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

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

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

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

    (1)创建配置文件。

    1. <bean id="testA" class="com.bean.TestA">     
    2.     <constructor-arg index="0" ref="testB"/>     
    3. </bean>     
    4. <bean id="testB" class="com.bean.TestB">     
    5.     <constructor-arg index="0" ref="testC"/>     
    6. </bean>     
    7. <bean id="testC" class="com.bean.TestC">     
    8.     <constructor-arg index="0" ref="testA"/>     
    9. </bean>     

    (2)创建测试用例。

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

0 0