一、简介
很多地方都会这样介绍自动装配:让Spring自动满足bean依赖,在满足依赖的过程中,会在Spring应用的上下文中寻找匹配某个bean需求的其他bean。也就是说,自动装配可以用来自动地创建bean,而且这个bean是满足依赖的。
二、基本用法
自动化装配Bean需要配置Spring核心依赖和Spring test依赖,如下所示:
<!--springframework-->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-core</artifactId>
<version>5.0.6.RELEASE</version>
</dependency>
<!--springdemo beans-->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-beans</artifactId>
<version>5.0.6.RELEASE</version>
</dependency>
<!--springdemo context-->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.0.6.RELEASE</version>
</dependency>
<!--springdemo test-->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-test</artifactId>
<version>5.0.6.RELEASE</version>
</dependency>
<!-- https://mvnrepository.com/artifact/junit/junit -->
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
</dependency>
1.在实现自动装配之前,我们要设置Bean可以被发现
①在类名上添加@Component注解,如
@Component
public class Tool {
}
②接着,我们需要新建一个空的配置类,用于扫描并创建Bean
//表明这是一个配置类
@Configuration
//启用组件扫描,默认扫描当前包
@ComponentScan
public class ScanConfig {
}
2.对于Spring来说,自动装配是通过@AutoWired注解实现的。但也可以使用JDK的@Inject注解,后者在非Spring框架中也能使用。
//这里的Person是一个声明了say方法的接口
@Component
public class SpiderMan implements Person {
private String name;
public SpiderMan(){
this.name = "SpiderMan";
}
//这里实现了依赖注入,因为Tool类上有@Component注解,所以这里可以把tool装进去
@Autowired
public SpiderMan(Tool tool){
this.name = "SpiderMan";
System.out.println("constructor param tool");
}
@Override
public String say() {
return "I am a spider man.";
}
}
3.测试
//RunWith标签用于自动创建应用上下文
@RunWith(SpringJUnit4ClassRunner.class)
//指定加载配置项
@ContextConfiguration(classes = ScanConfig.class)
public class VerifyAutoWired {
@Autowired
private Person person;
@Test
public void say(){
System.out.println(person.say());
}
}
三、进阶用法
像二中提到的,是自动化装配Bean的基础用法,一般来说,如果我们所要装配的接口只有一种实现的话,那么@AutoWired注解是可以完成自动装配这个工作的。但是如果像Person接口除了SpiderMan这种实现之外,如果还有其它的实现,这种做法就会报错。另外,自动化装配默认是单例模式(singleton),这在大部分情况下是合理的,但是某些情况下,对象会被污染,就不应该再使用单例模式了。
1.处理自动装配的歧义性
方法是在bean上添加@Primary注解或者使用@Qualifier限定符,将Spring可选的bean缩小到只有一个bean
//使用@Primary注解
@Component
@Primary
public class SpiderMan implements Person {
private String name;
@Autowired
public SpiderMan(Tool tool){
this.name = "SpiderMan";
System.out.println("constructor param tool");
}
@Override
public String say() {
return "I am a spider man.";
}
}
//使用@Qualifier
@Component("spiderman")//制定bean的id为spiderman
public class SpiderMan implements Person {
private String name;
@Autowired
public SpiderMan(Tool tool){
this.name = "SpiderMan";
System.out.println("constructor param tool");
}
@Override
public String say() {
return "I am a spider man.";
}
}
//使用@Qualifier注解
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes = ScanConfig.class)
public class VerifyAutoWired {
@Autowired
@Qualifier("spiderman")//注入有歧义的地方使用此注解
private Person person;
@Test
public void say(){
System.out.println(person.say());
}
}
2.设置bean作用域
Spring定义了多种作用域,可以基于这些作用域创建bean,包括:
- 单例(singleton):在整个应用中,只创建bean的一个实例。
- 原型(prototype):每次注入或者通过Spring应用上下文获取的时候,都会创建一个新的实例。
- 会话(session):在Web应用中,为每个会话创建一个bean实例。
- 请求(request):在Web应用中,为每个请求创建一个bean实例。
而单例则是默认的作用域,设置作用域的方法是在bean的类上添加@Scope注解
@Component("spiderman")
@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
public class SpiderMan implements Person {
private String name;
@Autowired
public SpiderMan(Tool tool){
this.name = "SpiderMan";
System.out.println("constructor param tool");
}
@Override
public String say() {
return "I am a spider man.";
}
}