仅有一个bean匹配所需的结果时,自动装配才是有效的。如果不仅有一个bean能够匹配结果的话,这种歧义性会阻碍Spring自动装配属性、构造器参数或方法参数。为了阐述自动装配的歧义性,假设我们使用@Autowired注解标注了setDessert()方法:

Spring实战之处理自动装配的歧义性-LMLPHP

在本例中,Dessert是一个接口,并且有三个类实现了这个接口,
分别为Cake、Cookies和IceCream:

Spring实战之处理自动装配的歧义性-LMLPHP

因为这三个实现均使用了@Component注解,在组件扫描的时候,能够发现它们并将其创建为Spring应用上下文里面的bean。然后,当Spring试图自动装配setDessert()中的Dessert参数时,它并没有唯一、无歧义的可选值。在从多种甜点中做出选择时,尽管大多数人并不会有什么困难,但是Spring却无法做出选择。Spring此时别无他法,只好宣告失败并抛出异常。更精确地讲,Spring会抛出

NoUniqueBeanDefinitionException

  当然,使用吃甜点的样例来阐述自动装配在遇到歧义性时所面临的问题多少有些牵强。在实际中,自动装配歧义性的问题其实比你想象中的更为罕见。就算这种歧义性确实是个问题,但更常见的情况是给定的类型只有一个实现类,因此自动装配能够很好地运行。但是,当确实发生歧义性的时候,Spring提供了多种可选方案来解决这样的问题。你可以将可选bean中的某一个设为首选(primary)的bean,或者使用限定符(qualifier)来帮助Spring将可选的bean的范围缩小到只有一个bean。

1.1标示首选的bean

如果你像我一样,喜欢所有类型的甜点,如蛋糕、饼干、冰激凌……它们都很美味。但如果只能在其中选择一种甜点的话,那你最喜欢的  是哪一种呢?假设冰激凌就是你最喜欢的甜点。在Spring中,可以通过@Primary来表达最喜欢的方案。@Primary能够与@Component组合用在组件扫描的bean上,也可以与@Bean组合用在Java配置的bean声明中。比如,下面的代码展现了如何将@Component注解的IceCream bean声明为首选的bean:

Spring实战之处理自动装配的歧义性-LMLPHP

Spring实战之处理自动装配的歧义性-LMLPHP如果你使用XML配置bean的话,同样可以实现这样的功能。<bean>元素有一个primary属性用来指定首选的bean:

Spring实战之处理自动装配的歧义性-LMLPHP

1.2限定自动装配的bean

  @Qualifier注解是使用限定符的主要方式。它可以与@Autowired和@Inject协同使用,在注入的时候指定想要注入进去的是哪个bean。例如,我们想要确保要将IceCream注入到setDessert()之

中:

Spring实战之处理自动装配的歧义性-LMLPHP

  这是使用限定符的最简单的例子。为@Qualifier注解所设置的参数就是想要注入的bean的ID。所有使用@Component注解声明的类都会创建为bean,并且bean的ID为首字母变为小写的类名。因此,@Qualifier("iceCream")指向的是组件扫描时所创建的bean,并且这个bean是IceCream类的实例。

04-28 12:15