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


90

我有一个要使用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)

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

Answers:


171

这是由于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;

9
您的生命更加安全,stackoverflow.com也是如此!:)
Rihards

5
如果可以的话,我给这十票。你是达曼,夫。
duffymo 2011年

3
许多人对此感到困惑,这意味着语义确实令人困惑。我不知道这种令人困惑的设计背后的原因是什么。
supertonsky 2012年

3
哇,我认为Spring是我最擅长的编程领域之一,直到今天我才从未遇到过这个问题,您为我节省了很多时间。谢谢!
Avi

2
关于如何使它与构造函数/方法注入一起使用的任何建议,例如@Resource不支持参数?
cjbooms

0

可能发生的另一件事是您正在自动装配bean的属性。在这种情况下,您无需对其进行自动装配,而只需创建setter方法并在bean定义中使用property标签(使用xml时)即可:

<bean id="cleaningUpOldFilesTasklet" class="com.example.mypackage.batch.tasklets.CleanUpOldFilesTasklet">
    <property name="directoriesToClean">
        <list>
            <value>asfs</value>
            <value>fvdvd</value>
            <value>sdfsfcc</value>
            <value>eeerer</value>
            <value>rerrer</value>
        </list>
    </property>
</bean>

和班级:

public class CleanUpOldFilesTasklet extends TransferingFilesTasklet implements Tasklet{

private long pastMillisForExpiration;
private final String dateFormat = "MM.dd";
Date currentDate = null;

List<String> directoriesToClean;

public void setDirectoriesToClean(List<String> directories){
    List<String> dirs = new ArrayList<>();
    for(String directory : directories){
        dirs.add(getSanitizedDir(directory));
    }
    this.directoriesToClean = dirs;
}

参见,@Autowired课堂上没有注释。

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.