首先我们先写一个自动装配的例子
//Dessert是指甜品
@Autowired
public void setDessert(Dessert dessert){
this.dessert=dessert;
}
如果Dessert是一个接口,他有如下的实现类
@Component 扫描这个注解的时候会创建bean,并且bean的ID为首字母为小写的类名
@Component
public class Cake implements Dessert{...}
@Component
public class Cookies implements Dessert{...}
@Component
public class IceCream implements Dessert{...}
这三个类都使用了@Component注解,所以当他们被扫描的时候能在spring应用上下文里面的bean,这时候要自动装配,就会发生歧义性,也就是自动装配不知道装配哪个bean。
那我们怎么解决这个歧义性呢?
- 我们可以通过设置首选(primary)
//这里是使用@Primary注解实现首选
@Component
@Primary
public class Cake implements Dessert{...}
@Bean会产生一个id为方法名的bean给spring容器
//java配置显式地声明
@Bean
@Primary
public Dessert cake(){
return new Cake();
}
//xml配置bean
<bean id="cake" class="com.spring.hjsjy">
primary="true"/>
这里面也会有一个问题,如果我们不止一个primary那么首选就没有办法解决歧义性,我们可以通过下面的方法解决。
- 使用限定符(qualifier)
//Dessert是指甜品
@Autowired
@Qualifier("cake")//限定符
public void setDessert(Dessert dessert){
this.dessert=dessert;
}
这时也会有个问题,如果我们重构了Cake类,把名字重命名为BigCake类,方法的限定修饰符和要注入的bean名字紧耦合。这时候就无法和限定符匹配。
解决办法:创建自定义的限定符
@Component
@Qualifier("hjsjy")
public class Cake implements Dessert{...}
//Dessert是指甜品
@Autowired
@Qualifier("hjsjy")
public void setDessert(Dessert dessert){
this.dessert=dessert;
}
这样的话我们随便重构Cake类的都不会影响自动装配。
本文参考资料:《spring实战(第四版)》