IoC容器
IoC容器本身利用了DI依赖倒转,通过构建Bean类和配置XML文件等方式来通过IoC容器生成Bean(这也是依赖倒转的由来,与手动生成不同,交由容器生成),并通过ApplicationContext来获取这些Bean。
Bean生命周期
原始的Bean生命周期如下
Bean定义 -> Initial初始化 -> 生存 -> Destroy销毁
其中Initial可以通过XML中的init-method来定义,而Destroy也可以通过destroy-method定义(需要注意的是,使用这种方法配置的method必须是void返回值且无参的);或者可以使Bean类实现InitializationBean接口,以及DestryBean接口来进行方法构建。
需要注意的是,Destroy必须使用AbstractApplicationContext.registerShutdownHook()来确保正常关闭才会被调用。
另外,通过实现BeanPostProcessor后置处理器接口,可以为Bean的加载额外添加两个步骤。
Bean定义 -> postProcessBeforeInitialization后置处理前初始化 -> Initial初始化 -> postProcessAfterInitialization后置处理后初始化 -> 生存 -> Destroy销毁
Bean继承
对于Bean,也存在继承选项,在xml中可以指定parent,并直接得到父Bean的所有配置项。需要注意的是,如果子Bean中不包含所有的父Bean的配置项,则会抛出异常。
另外,可以额外定义Bean模板,他们不指定class,相对的,需要将abstract指定为true。在没有子Bean继承父Bean时,将会产生错误。
<bean id="helloChild" class="beans.HelloChild" parent="helloParent">
<property name="childCode" value="2"/>
</bean>
<!-- 抽象父Bean,可以不指定class -->
<bean id="helloParent" abstract="true">
<property name="msg" value="HelloParent!"/>
</bean>
对应的Java类
package beans;
public class HelloChild {
private String msg;
private int childCode;
public String getMsg() {
return msg;
}
public void setMsg(String msg) {
this.msg = msg;
}
public int getChildCode() {
return childCode;
}
public void setChildCode(int childCode) {
this.childCode = childCode;
}
}
依赖注入
依赖注入即把BeanA所依赖的BeanB通过方法注入到BeanA中,而不是直接在BeanA中创建一个BeanB的实例,通过依赖注入可以将实现与用法分离,能够使得这些实例更加独立。
相较于一种写法,依赖注入更像是一种思想,通过一些特殊的实例构造方式来解耦Bean之间的依赖。
基于构造函数的依赖注入
在A构造时就注入B作为成员变量,需要在XML中配置<constructor-arg index=0 ref="BeanB"/>
,加入index则是为了保证参数传递的顺序性,防止参数传递出错。
基于设值函数的依赖注入
相对于构造函数,设值函数的使用更加泛化,在XML中配置<property name="BeanB" ref="BeanB"/>
即可,相对于常规的Bean定义赋值,只是将value标签部分替换为了ref(reference引用)。
内部Beans注入
即使需要在BeanA内部创建一个BeanB的实例,仍可以用依赖注入的方案解决。而使用内部Beans注入的XML配置则不需要BeanId(类似于匿名内部类,该Bean仅作用于BeanA中)。
<bean id="heaven" class="beans.HeavenBean">
<property name="angel">
<!-- 直接创建一个AngelBean的实例 -->
<bean class="beans.AngelBean">
<property name="name" value="Panty"/>
</bean>
</property>
</bean>
集合注入
在需要向BeanA中的容器中注入内容时,就需要用到几个额外的特殊标签;在这里举一个Map的例子。
<bean id="heavenS" class="beans.HeavenBean">
<property name="angelMap">
<map>
<entry key="Stock" value-ref="angel"/>
<entry key="Panty" value-ref="angelS"/>
</map>
</property>
</bean>