spring初始化bean的顺序

问题

不是说单个bean的初始化顺序。这个顺序大体上是构造方法-set方法-init方法,详细的可以百度。

这里说的是在spring容器中互不相关的两个bean的初始化顺序。例如:

<bean id="a" class="A"/>

<bean id="b" class="B"/>

经过昨天遇到的问题,现在知道了这两个bean的顺序是以xml文件中的顺序为准的。

例如a配置在b的前面,那么spring就先装配实例a,然后再装配实例b。

题外,其实ibatis、mybatis里,sql和include等标签也是一样,sql必须定义在include的前面。

回过来说昨天遇到的问题。简单说是这样的:

class A{
    private X x=B.service();
}

class  B{
    private static C c;
    public static X service(){
        // TAG
        c.yyyy();
    }
   public void setC(C c){
       B.c=c;
    }
}

配置是:

<bean id="a" class="A"/>
<bean id="b" class="B">
    <property name="c" ref="xxxxx"/>
</bean>

A通过B的静态方法service(),在构造方法中去初始化成员变量x;B的静态方法service()需要调用类变量c,而c是spring通过成员方法setC(C c)来注入的。

这种情况下,如果spring配置中,a在b的前面,则a在初始化的时候,会在TAG处抛出空指针异常。因为这个时候,srping还没有完成b的装配,因此B.c是null,调用c.yyyy()当然会抛异常。

解决

严格来说,类C不应该声明为类B的静态变量。另外一篇文章里我应该也提到过:有业务含义的东西,就不要声明为静态方法。变量也是如此