三、(三)自动装配
自动装配:有助于减少甚至消除<property>元素和<constructor-arg>元素,让Spring自动识别如何装配Bean的依赖关系。
自动装配
Spring提供了四种自动装配策略:
byName
——把与Bean的属性与具有相同名字或者ID的其他Bean自动装配到Bean的对应属性中。如果没有跟属性的名字相匹配的Bean,则该属性不进行装配。
byType
——把与Bean的属性与具有相同类型的其他Bean自动装配到Bean的对应的属性中。如果没有跟属性的类型相匹配的Bean,则该属性不进行装配。
constructor
——把与Bean的构造器入参具有相同类型的其他Bean自动装配到Bean的构造器的对应入参中。
autodetect
——首先尝试使用constructor进行自动装配。如果失败,再尝试使用byType进行自动装配。
byName
<bean name="student" class="pray.wang.xml.entity.Person">
<property name="score">
<props>
<prop key="数学">80</prop>
<prop key="英语">90</prop>
<prop key="语文">95</prop>
</props>
</property>
</bean>
<bean name="peopleService" class="pray.wang.xml.service.StudentService" autowire="byName">
</bean>
byType
<bean name="student" class="pray.wang.xml.entity.Person">
<property name="score">
<props>
<prop key="数学">80</prop>
<prop key="英语">90</prop>
<prop key="语文">95</prop>
</props>
</property>
</bean>
<bean name="peopleService" class="pray.wang.xml.service.StudentService" autowire="byType">
</bean>
constructor
<bean name="student" class="pray.wang.xml.entity.Person">
<property name="score">
<props>
<prop key="数学">80</prop>
<prop key="英语">90</prop>
<prop key="语文">95</prop>
</props>
</property>
</bean>
<bean name="peopleService" class="pray.wang.xml.service.StudentService" autowire="constructor">
</bean>
autodetect
<bean name="student" class="pray.wang.xml.entity.Person">
<property name="score">
<props>
<prop key="数学">80</prop>
<prop key="英语">90</prop>
<prop key="语文">95</prop>
</props>
</property>
</bean>
<bean name="peopleService" class="pray.wang.xml.service.StudentService" autowire="autodetect">
</bean>
通过byType自动装配存在一个局限性:如果Spring寻找到多个Bean,它们的类型与需要自动装配的属性的类型都相匹配。在这种场景下,Spring不会猜测哪一个Bean更适合自动装配,而是选择抛出异常。所以,应用只允许存在一个Bean与需要自动装配的属性类型相匹配。
为了避免因使用byType自动装配而带来的歧义,Spring提供了另外两种选择:可以为自动装配标识一个首选Bean,或者可以消除某个Bean自动装配的候选资格。
- 为自动装配标识一个首选Bean,可以使用<bean>元素的primary属性。如果只有一个自动装配的候选Bean的primary属性设置为true,那么该Bean将比其他候选Bean优先被选择。
<bean name="student" class="pray.wang.xml.entity.Person" primary="true">
<property name="score">
<props>
<prop key="数学">80</prop>
<prop key="英语">90</prop>
<prop key="语文">95</prop>
</props>
</property>
</bean>
<bean name="teacher" class="pray.wang.xml.entity.Person">
<property name="score">
<props>
<prop key="数学">100</prop>
<prop key="英语">100</prop>
<prop key="语文">100</prop>
</props>
</property>
</bean>
<bean name="peopleService" class="pray.wang.xml.service.StudentService" autowire="byType">
</bean>
- 消除某个Bean自动装配的候选资格,可以设置这些Bean的autowire-candidate属性为false。
<bean name="student" class="pray.wang.xml.entity.Person">
<property name="score">
<props>
<prop key="数学">80</prop>
<prop key="英语">90</prop>
<prop key="语文">95</prop>
</props>
</property>
</bean>
<bean name="teacher" class="pray.wang.xml.entity.Person" autowire-candidate="false">
<property name="score">
<props>
<prop key="数学">100</prop>
<prop key="英语">100</prop>
<prop key="语文">100</prop>
</props>
</property>
</bean>
<bean name="peopleService" class="pray.wang.xml.service.StudentService" autowire="byType">
</bean>
默认自动装配
如果需要为Spring上下文中所有的Bean配置相同的autowire属性,可以在根元素<beans>上增加一个default-autowire属性。
当然你也可以在一个Spring应用上下文中定义多个配置文件,每一个配置文件都有他自己的默认自动装配策略。
同样,不能因为我们配置了一个默认的自动装配策略,就意味着所有的Bean都只能使用这个默认的自动装配策略。我们还可以使用<bean>元素的autowire属性来覆盖<beans>元素所配置的默认自动装配策略。
混合使用自动装配和显示装配
我们对某个Bean选择了自动装配策略,并不代表我们不能对该Bean的某些属性进行显示装配。特别的,当使用byType自动装配策略所可能产生的装配不确定性问题导致Spring抛出异常,通过显示的装配来覆盖自动装配
使用注解装配
使用注解装配比起使用XML允许更细粒度的自动装配,我们可以选择性的标注一个属性来对其应用自动装配。
Spring容器默认禁用注解装配。所以在基于注解的自动装配前,我们需要在Spring中启用它。
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-4.0.xsd
">
<context:annotation-config></context:annotation-config>
</beans>
Spring支持三种用于自动装配的注解:
- Spring自带的@Autowired注解
- JSR-330的@Inject注解
- JSR-250的@Resource注解
@Autowired
@Autowired注解不仅可以标注setter方法,还可以标注需要自动装配Bean引用的任意方法,甚至可以标注构造器。另外,我们还可以用它来直接标注属性,并删除setter方法,甚至不受限于私有属性。
-
可选的自动装配
默认情况下,@Autowired注解具有强契约特性,其标注的属性或者参数必须是可装配的。如果没有Bean可以装配到@Autowired注解所标注的属性或参数中,自动装配就会失败并抛出异常。为此,我们可以通过设置@Autowired的required属性为false来配置自动装配是可选的。 -
限定歧义性的依赖
如果Spring并不缺少适合自动装配的Bean,可能有两个或者多个Bean都能满足装配条件,并且都可以被装配到属性或者参数中。在这种场景下,@Autowired注解没有办法选择哪一个Bean才是它真正需要的,所以也会导致自动装配失败并抛出异常。为此,我们可以配合使用@Qualifier注解来缩小自动装配挑选候选Bean的范围。
@Inject
@Inject注解几乎可以完全替换Spring的@Autowired注解。和@Autowired一样,@Inject可以用来自动装配属性,方法和构造器;不同的是,@Inject没有required属性。因此,@Inject注解所标注的依赖关系必须存在,否则抛出异常。
- 限定歧义性的依赖
像@Autowired一样,@Inject注解也会导致歧义性的Bean定义。相对于@Autowired所对应的@Qualifier注解,@Inject所对应的是@Named注解。
@Qualifier与@Named的关键区别在于语义层面。@Qualifier注解帮助我们缩小所匹配Bean的选择范围,而@Qualifier通过Bean的ID来标识可选择的Bean。
创建自定义的限定器
除了Spring的@Qualifier注解,JSR-330在javax.inject包里也有自己@Qualifier注解。不像Spring的@Qualifier,JSR-330不建议我们使用该注解。相反,JSR-330鼓励我们使用该注解来创建自定义的限定器注解。事实上,无论是Spring的@Qualifier,还是JSR-330的@Qualifier注解都可以创建自定义的限定器注解,用来缩小自动装配挑选候选Bean的范围。
@Target({ElementType.CONSTRUCTOR,ElementType.FIELD,ElementType.METHOD,ElementType.PARAMETER,ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Qualifier
public @interface FirstSelected {
}