一尘不染

使用util模式自动连接列表会产生NoSuchBeanDefinitionException

spring

我有一个要使用Spring util命名空间注入命名列表的bean, <util:list id="myList">但是Spring正在寻找String类型的bean的集合。我坏的测试是:

@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)

令我困惑的是,我认为这将是预期的工作方式。


阅读 365

收藏
2020-04-12

共1个答案

一尘不染

这是由于3.11.2中指定的@Autowired行为的一个相当模糊的部分。@Autowired

ApplicationContext通过将注释添加到需要该类型数组的字段或方法中,还可以提供中特定类型的所有bean 。

同样适用于类型化集合…

换句话说,通过说@Autowired @Qualifier("myList") List<String>,你实际上是在要求“给我java.lang.String具有限定符“ myList” 的所有类型的bean的列表。

该解决方案在3.11.3中提到。使用限定符对基于注释的自动装配进行微调:

如果你打算通过名称表示注释驱动的注入,则不要主要使用@Autowired-即使从技术上讲,它可以通过@Qualifier 值来引用Bean名称。相反,最好使用JSR-250 @Resource 批注,该批注在语义上定义为通过其唯一名称标识特定目标组件,而声明的类型与匹配过程无关。

由于这种语义差异的特定结果,本身无法定义为集合或映射类型的bean无法注入,@Autowired因为类型匹配不适用于它们。使用 @Resource的唯一名称等豆类,指的是特定集合/图豆。

因此,在测试中使用它,即可正常工作:

@Resource(name="myList") private List<String> stringList;
2020-04-12