前言
我们在开发Spring应用时可能会不小心注入两个相同类型的Bean,比如实现了两个相同Service接口的类,示例伪代码如下:
1 | interface SampleService { |
这时候我们用SampleService接口注入
1 |
|
启动应用后,Spring就会优雅地提示如下错误:
Exception in thread “main” org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type ‘com.john.primary.SampleService’ available: expected single matching bean but found 2: ServiceA,ServiceB
但是我们不想报错且想获取其中某一个Bean,这时候我们该怎么办呢?
解决方案
既然包含了两个相同类型的Bean,通常来说我们只要把其中一个Bean不注入就好,那如果我们想保留这两个相同类型的Bean,但是又想让SampleService正常注入呢?
如果我们是用早期Spring的Xml配置Bean时,可以使用如下两种方式解决:
那么我们可以在其中一个Bean配置里加上autowire-candidate=”false”
1
2<bean id="serviceA" class="com.john.primary.ServiceA" />
<bean id="serviceB" class="com.john.primary.ServiceB" autowire-candidate="false" />或者在其中一个Bean配置里加上primary=”true”:
1
2<bean id="serviceA" class="com.john.primary.ServiceA" primary="true"/>
<bean id="serviceB" class="com.john.primary.ServiceB" />采用javax.annotation.Priority注解
这种方式需要我们在BeanFactory里加上dependencyComparator,示例代码如下:
1
2
3
4DefaultListableBeanFactory beanFactory = (DefaultListableBeanFactory)context.getBeanFactory();
//@Priority注解比较
beanFactory.setDependencyComparator(AnnotationAwareOrderComparator.INSTANCE);
SampleService sampleService= context.getBean(SampleService.class);实现注解Order或者实现org.springframework.core.Ordered接口
1
2
3
4
5
6
7
8
9
10
11
12
13
14//@Order
public class ServiceA implements SampleService,Ordered {
public int getOrder() {
return 0;
}
public String toString() {
return "ServiceA{}";
}
}这种方式需要我们重写AnnotationAwareOrderComparator的getPriority方法,示例代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16public class PriorityOrderComparator extends AnnotationAwareOrderComparator {
/**
* Shared default instance of {@code PriorityOrderComparator}.
*/
public static final PriorityOrderComparator INSTANCE = new PriorityOrderComparator();
public Integer getPriority(Object obj) {
//先获取Priority
Integer order = super.getPriority(obj);
if(order == null)
//获取Order注解或者Ordered接口返回值
return super.findOrder(obj);
return order;
}
}
我们还可以使用目前流行的注解方式来实现,Spring文档中也提到过:
Because autowiring by type may lead to multiple candidates, it is often necessary to have more control over the selection process. One way to accomplish this is with Spring’s @Primary annotation. @Primary indicates that a particular bean should be given preference when multiple beans are candidates to be autowired to a single-valued dependency. If exactly one primary bean exists among the candidates, it becomes the autowired value.
那么可以使用如下方式:
@Primary注解:
该注解可以标注在类上或者方法上,示例如下:
1
2
3
4
5
6
7
class ServiceA implements SampleService{
String getName(){
return "john";
}
}注解在有@Bean注解的方法上:
1
2
3
4
5
SampleService sampleService(){
return new ServiceA();
}还是采用Xml配置中的第三或者第四种解决方案,只是采用第四种方案的话还是需要重新扩展AnnotationAwareOrderComparator
总结
这篇文章介绍了解决org.springframework.beans.factory.NoUniqueBeanDefinitionException异常的一些解决方案,从这些解决方案可以看出Spring框架的设计精妙,留给我们开发者的扩展点实在很多了。下篇文章我们来了解下这些解决方案的实现原理,如果你也有其他的解决方案,请留言告诉我。