Bean的自动装配
自动装配(autowiring)有助于减少甚至消除配置<property>元素和<constructor-arg>元素,让Spring自动识别如何装配Bean的依赖关系。
1.1 自动装配Bean属性:
Spring提供了四种各具特色的自动装配策略:
1、byName-把与Bean的属性具有相同名字的其他Bean自动装配到Bean的对应属性中。如果没有跟属性的名字相匹配的Bean,则该属性不进行装配。
2、byType-把与Bean的属性具有相同类型的其他Bean自动装配到Bean的对应属性中。如果没有跟属性的类型相匹配的Bean,则该属性不被装配。
3、constructor-把与Bean的构造器入参具有相同类型的其他Bean自动装配到Bean构造器的对应入参中。
4、autodetect-首先尝试使用constructor进行自动装配。如果失败,再尝试使用byType进行自动装配。
1.1.1 byName自动装配:
不使用byName自动装配之前是这样声明的:
<bean id="kenny2" class="com.springinaction.springidol.Instrumentalist"> <property name="song" value="Jingle Bells"></property> <property name="instrument" ref="instrument"></property> </bean>
使用byName自动装配之后是这样的:
<bean id="instrument" class="com.springinaction.springidol.Saxophone"/> <bean id="kenny2" class="com.springinaction.springidol.Instrumentalist" autowire="byName"> <property name="song" value="Jingle Bells"></property> </bean>
byName自动装配遵循一项约定,为属性自动装配ID与该属性的名字相同的Bean。通过设置autowire属性为byName,spring将为该bean的属性寻找与其名字相同的Spring Bean。使用byName的缺点是需要Bean的名字与其他bean的属性的名字一样。
1.1.2 byType自动装配:
byType自动装配的工作方式类似于byName自动装配,只不过不再是匹配属性的名字而是检查属性的类型。当我们尝试使用byType自动装配时,Spring会寻找哪一个Bean的类型与属性的类型相匹配。
但是byType自动装配存在一个局限性:如果Spring寻找到多个Bean,它们的类型与需要自动装配的属性的类型都相匹配,怎么办呢?在这种场景下,Spring不会猜测哪一个Bean更适合自动装配,而是选择抛出异常。所以,应用只允许存在一个Bean与需要自动装配的属性类型相匹配。
为了避免因为使用byType自动装配而带来的歧义,Spring为我们提供了另外两种选择:可以为自动装配标识一个首选的Bean,或者可以取消某个Bean自动装配的候选资格。
为自动装配标识一个首选Bean,可以使用<bean>元素的primary属性。如果只有一个自动装配的候选Bean的primary属性设为true,那么该Bean将比其他候选Bean优先被选择。
但是primary属性有个很怪异的一点:它默认设置为true。所以,为了使用primary属性,我们不得不将所有非首选的primary属性设为false。例如:
<bean id="instrument" class="com.springinaction.springidol.Saxophone" primary="false"/>
primary属性仅对标识首选Bean有意义。如果在自动装配时,我们希望排除某些Bean,那可以设置这些Bean的autowire-candidate属性为false,如下所示:
<bean id="instrument" class="com.springinaction.springidol.Saxophone" autowire-candidate="false"/>
1.1.3 constructor自动装配:
<bean id="duke" class="com.springinaction.springidol.PoeticJuggler" autowire="constructor" />
constructor自动装配具有和byType自动装配相同的局限性。当发现多个Bean匹配某个构造器的入参时,Spring不会尝试猜测哪一个Bean更适合自动装配。此外,如果一个类有多个构造器,它们都满足自动装配的条件时,Spring也不会尝试猜测哪一个构造器更适合使用。
1.1.4 最佳自动装配
如果想要自动装配bean,但是又不能决定该使用哪一种类型的自动装配,那么可以设置autowire属性为autodetect,由spring 来决定。
<bean id="duke" class="com.springinaction.springidol.PoeticJuggler" autowire="autodetect" />
当配置一个Bean的autowire属性为autodetect时,Spring将首先使用constructor自动装配,如果没有发现与构造器相匹配的Bean时,Spring将尝试使用byType自动装配。
1.2 默认自动装配
如果需要为Spring应用上下文中的每一个bean(或者其中的大多数)配置相同的autowire属性,那么就可以要求Spring为它所创建的所有bean应用相同的自动装配策略来简化配置,所需要的做的仅仅是在根元素<beans>上添加一个default-autowire属性:
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd" default-autowire="byType"> </beans>
默认情况下,default-autowire属性被设置为none,标示所有Bean都不使用自动装配,除非bean自己配置了autowire属性。在某个配置文件中的default-autowire只是应用于指定Spring配置文件中的所有Bean,并没有应用于Spring应用上下文中的所有bean。可以在一个Spring应用上下文中定义多个配置文件,每一个配置文件都可以有自己的默认自动装配策略。同样的,不能因为配置了一个默认的自动装配策略,就意味着所有的bean都只能使用这个默认的自动装配策略,还可以使用<bean>元素的autowire属性来覆盖<bean>元素所配置的默认自动装配策略。
1.3 混合使用自动装配和显示装配
如果对某个bean选择了自动装配,并不代表不能对该bean的某些属性进行显示装配。我们仍然可以为任意一个属性配置<property>元素,就像我们之前没有配置autowire一样。
<bean id="kenny" class="com.springinaction.springidol.Instrumentalist" autowire="byType"> <property name="song" value="Jingle Bells" /> <property name="instrument" ref="saxophone" /> </bean>
混合使用自动装配和手工装配非常成功的解决了使用byType自动装配策略所可能产生的装配不确定性问题。如果在Spring上下文中有多个bean实现了Instrument接口,为了规避Spring无法从这几个实现了Instrument接口的Bean中进行明确挑选而抛出异常,可以显示的装配instrument属性来覆盖自动装配。
使用混合装配的最后一个注意事项:当使用constructor自动装配策略时,必须让Spring自动装配构造器的所有入参,而不能混合使用constructor自动装配策略和<constructor-arg>元素。