自动装配(autowiring)有助于减少甚至消除配置<property>元素和<constructor-arg>元素,让Spring自动识别如何装配Bean的依赖关系。
自动检测(autodiscovery)比自动装配更进了一步,让Spring能够自动识别哪些类需要被配置成Spring Bean,从而减少对<bean>元素的使用。
1.自动装配属性
1.1 4种类型的自动装配
● byName——把与Bean的属性具有相同名字(或者ID)的其他Bean自动装配到Bean的对应属性中。如果没有跟属性的名字相匹配的Bean,则该属性不进行装配。● byType——把与Bean的属性具有相同类型的其他Bean自动装配到Bean的对应属性中。如果没有跟属性的类型相匹配的Bean,则该属性不被装配。
● constructor——把与Bean的构造器入参具有相同类型的其他Bean自动装配到Bean构造器的对应入参中。
● autodetect——首先尝试使用constructor进行自动装配。如果失败,在尝试使用byType进行自动装配。
byName自动装配:
<bean id = "kenny" class = "com.ouc.test.springs.Instruments" autowire = "byName" >
<property name = "song" value = "bells" />
</bean>
byType自动装配:
如果Spring寻找到多个Bean,它们的类型与自动装配的属性类型都相匹配,Spring不会猜测哪一个Bean更适合自动装配,而是会抛出异常。
可以为自动装配标识一个首选Bean,或者可以取消某个Bean自动装配的候选资格。为了使用primary属性,不得不将非首选Bean的primary属性设置为false。
可以为自动装配标识一个首选Bean,或者可以取消某个Bean自动装配的候选资格。为了使用primary属性,不得不将非首选Bean的primary属性设置为false。
<bean id = "saxophone" class = "com.ouc.test.springs.Saxophone" primary = "false" />
如果希望排除某些Bean,可以设置这些Bean的autowire-candidate属性为false。
<bean id = "saxophone" class = "com.ouc.test.springs.Saxophone" autowire-candidate = "false" />
constructor自动装配:
如果要通过构造器注入来配置Bean,可以移除<constructor-arg>元素,由Spring在应用上下文中自动选择Bean注入到构造器入参中。
<bean id = "saxophone" class="com.ouc.springs.test.Saxophone" autowire ="constructor" />
最佳自动装配:
<bean id = "saxophone" class="com.ouc.springs.test.Saxophone" autowire ="autodetect" />
1.2 默认自动装配
<?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.1.xsd default-autowire="byType" > </beans>
2 .使用注解装配
Spring容器默认禁用注解装配。启用注解装配最简单的方式是使用Spring的context命名空间配置中的<context:annotation-config>元素。
Spring 3 支持几种不同的用于自动装配的注解:
● Spring自带的@Autowired注解;
● JSR-330的@Inject注解;
● JSR-250的@Resource注解。
2.1 使用@Autowired
@Autowired
public void setInstrument(Instrument instrument){
this.instrument = instrument;
}
可以使用@Autowired注解直接标注属性,并删除setter方法:
@Autowired private Instrument instrument;
默认情况下,@Autowired所标注的属性或参数必须是可以装配的。如果没有Bean可以装配到@Autowired所标注的属性或参数中,自动装配就会失败(抛出NoSuchBeanDefinitionException).
可以通过设置@Autowired的required属性为false来配置自动装配是可选的。
@Autowired(required=false) private Instrument instrument;
@Qualifier注解缩小了自动装配挑选候选Bean的范围,通过指定Bean的ID把选择范围缩小到只剩下一个Bean。
@Autowired @Qualifier("guitar") private Instrument instrument;
import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; import org.springframework.beans.factory.annotation.Qualifier; @Target({ElementType.FIELD, ElementType.PARAMETER, ElementType.TYPE}) @Retention(RetentionPolicy.RUNTIME) @Qualifier public @interface StringedInstrument{}
2.2 借助@Inject实现基于标准的自动装配
@Inject private Instrument instrument;
限定@Inject所标注的属性。
@Inject @Named("guitar") private Instrument instrument;
创建自定义的JSR-330 Qualifier
import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; import javax.inject.Qualifier; @Target({ElementType.FIELD, ElementType.PARAMETER, ElementType.TYPE}) @Retention(RetentionPolicy.RUNTIME) @Qualifier public @interface StringedInstrument{}
2.3 在注解注入中使用表达式
@Value("#{systemProperties.myFavoriteSong}") private String song;
3.自动检测Bean
使用<context:component-scan>元素配置自动检测。<?xml version="1.0" encoding="UTF-8"?> <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-3.1.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/beans/spring-context-3.1.xsd" > <context:component-scan base-package="com.ouc.springs.test" > </context:component-scan> </beans>
3.1为自动检测标注Bean
默认情况下,<context:component-scan>查找使用构造型(stereotype)注解所标注的类,这些特殊的注解如下:@Component——通用的构造型注解,标识该类为Spring组件。
@Controller——标识将该类定义为Spring MVC Controller。
@Repository——标识将该类定义为数据仓库。
@Service——标识将该类定义为服务。
使用@Component标注的任意自定义注解。
3.2 过滤组件扫描
通过为<context:component-scan>配置<context:include-filter>和<context:exclude-filter>子元素,可以随意调整扫描行为。<context:component-scan base-package="com.ouc.springs.test" >
<context:include-filter type="assignable" expression="com.ouc.springs.tst.Instrument" />
<context:exclude-filter type="annotation" expression="com.ouc.springs.tst.SkipIt" />
</context:component-scan>
4.使用Spring基于Java的配置
4.1 定义一个配置类
import org.springframework.context.annotation.Configuration; @Configuration public class SpringIdolConfig{}
4.2 声明一个简单的Bean
@Bean public Performer duke(){ return new Juggler(); }