我有一个要使用Spring util命名空间注入命名列表的bean, <util:list id="myList">但是Spring正在寻找String类型的bean的集合。我坏的测试是:
<util:list id="myList">
@RunWith(SpringJUnit4ClassRunner.class) @ContextConfiguration public class ListInjectionTest { @Autowired @Qualifier("myList") private List<String> stringList; @Test public void testNotNull() { TestCase.assertNotNull("stringList not null", stringList); } }
我的上下文是:
<beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:util="http://www.springframework.org/schema/util" xmlns="http://www.springframework.org/schema/beans" xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util-2.5.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd"> <util:list id="myList"> <value>foo</value> <value>bar</value> </util:list> </beans>
但是我明白了
Caused by: org.springframework.beans.factory.NoSuchBeanDefinitionException: No matching bean of type [java.lang.String] found for dependency [collection of java.lang.String]: expected at least 1 bean which qualifies as autowire candidate for this dependency. Dependency annotations: {@org.springframework.beans.factory.annotation.Autowired(required=true), @org.springframework.beans.factory.annotation.Qualifier(value=myList)} at org.springframework.beans.factory.support.DefaultListableBeanFactory.raiseNoSuchBeanDefinitionException(DefaultListableBeanFactory.java:726) at org.springframework.beans.factory.support.DefaultListableBeanFactory.resolveDependency(DefaultListableBeanFactory.java:571) at org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor$AutowiredFieldElement.inject(AutowiredAnnotationBeanPostProcessor.java:412)
令我困惑的是,我认为这将是预期的工作方式。
这是由于3.11.2中指定的@Autowired行为的一个相当模糊的部分。@Autowired:
@Autowired
ApplicationContext通过将注释添加到需要该类型数组的字段或方法中,还可以提供中特定类型的所有bean 。
ApplicationContext
同样适用于类型化集合…
换句话说,通过说@Autowired @Qualifier("myList") List<String>,你实际上是在要求“给我java.lang.String具有限定符“ myList” 的所有类型的bean的列表。
@Autowired @Qualifier("myList") List<String>
该解决方案在3.11.3中提到。使用限定符对基于注释的自动装配进行微调:
如果你打算通过名称表示注释驱动的注入,则不要主要使用@Autowired-即使从技术上讲,它可以通过@Qualifier 值来引用Bean名称。相反,最好使用JSR-250 @Resource 批注,该批注在语义上定义为通过其唯一名称标识特定目标组件,而声明的类型与匹配过程无关。
JSR-250 @Resource
由于这种语义差异的特定结果,本身无法定义为集合或映射类型的bean无法注入,@Autowired因为类型匹配不适用于它们。使用 @Resource的唯一名称等豆类,指的是特定集合/图豆。
@Resource
因此,在测试中使用它,即可正常工作:
@Resource(name="myList") private List<String> stringList;