Spring容器可以自动连接协作bean之间的关系。您可以通过检查ApplicationContext的内容让Spring自动为您的bean解析协作者(其他bean)。自动装配具有以下优点:
- 自动装配可以显着减少对指定属性或构造函数参数的需要。(在本章其他地方讨论的其他机制,如bean模板,在这方面也很有价值。)
- 自动装配可以随着对象的发展更新配置。例如,如果需要向类添加依赖项,则可以自动满足该依赖项,而无需修改配置。因此,自动装配在开发期间尤其有用,而不会在代码库变得更稳定时否定切换到显式装配的选项。
1. XML自动装配
使用基于XML的配置元数据(请参阅依赖注入)时,可以使用<bean />元素的autowire属性为bean定义指定autowire模式。自动装配功能有四种模式。下表描述了四种自动装配模式:
Mode | Explanation |
---|---|
no | (默认)无自动装配。Bean引用必须由ref元素定义。不建议对较大的部署更改默认设置,因为明确指定协作者可以提供更好的控制和清晰度。在某种程度上,它记录了系统的结构。(任然可以使用@Autowried 注解自动装配) |
byName | 按属性名称自动装配。 Spring查找与需要自动装配的属性同名的bean。例如,如果bean定义按名称设置为autowire并且它包含master属性(即,它具有setMaster(..)方法),则Spring会查找名为master的bean定义并使用它来设置属性。如果未找到也不回报错,这在运行时可能会报空指针异常。 |
byType | 如果容器中只存在一个属性类型的bean,则允许属性自动装配。如果存在多个,则抛出致命异常,这表示您不能对该bean使用byType自动装配。如果没有匹配的bean,则不会发生任何事情(该属性未设置)。如果未找到也不回报错,这在运行时可能会报空指针异常。 |
constructor | 类似于byType但适用于构造函数参数。如果容器中没有构造函数参数类型的一个bean,则会引发致命错误。 |
通过以上我们可以知道,如果你的依赖是必须依赖(一定要注入),则不要使用byType,byName的方式自动装配,而是使用构造方法自动装配。因为构造方法自动装配,会在启动时就会报错(找不到bean)。
org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type
2. 注解的自动装配
使用注解的自动装配,自动装配的颗粒度更细。我们可以使用@AutoWried
,@Resource
(JSR-250),@Inject(需要javax.inject
包)
和xml不一样的是基于注解的自动装配,如果找不到bean,会在启动时报错。而且@AutoWried
可以通过它的required
属性来控制是否需要注入。
- @AutoWried
注入时使用在field,构造器,方法。
一般情况下是基于类型自动装配。偶尔也可以通过名称(不使用qualifier和primary)。
有一种特殊情况,如果有两个相同类型的候选bean,Spring会将注入点名称(即field名称或参数名称)与目标bean名称进行匹配,然后选择同名的候选人,如果有的话。
- @Resource
注入时使用在field和方法上(只支持单个参数)。
默认是基于名称自动装配的。@Resource
自动装配按以下顺序装配:
- 使用
@Resource
的name
属性,例如:@Resource(name = "Bean1"); - 不使用
name
属性,如果是加在field上Spring会根据field的名称匹配,加在方法上Spring会根据set方法的名称或set方法的参数名来匹配,例如,Bean的名称是myBeanName,则以下示例都能成功匹配:
@Resource
private MyBeanName myBeanName;
-----------------------------------------------------------------------
@Resource
private MyBeanName myBeanName22;//根据类型匹配和@Autowried一样
-----------------------------------------------------------------------
private MyBeanName myBeanName13;
@Resource
public void setMyBeanName(MyBeanName myBeanName13){
this.mybeanName = myBeanName13;
}
---------------------------------------------------------------------
private MyBeanName myBeanName;
@Resource
public void setMyBeanName33(MyBeanName myBeanName){
this.mybeanName = myBeanName;
}
- 如果以上条件都不满足,则根据类型匹配,如果有多个或没有,则报错。
从JDK9开始,@Resource
不包含再jdk中,使用时需要导入java.xml.ws.annotation
模块。
- @Inject
和@Autowried
一样,该注解可以使用在field,构造器,方法上。
通过类型注入。
有一点需要注意,如果你的装配使用注解装配,则会忽略你的xml如何配置的
autowire
属性,也就是说无论> 你怎么配置autowire
都不起作用。还有如果你想使用xml的注解自动装配,需要添加
<context:annotation-config/>