使用 util 架构自动连接列表会给 NoSuchBeanDefinitionException

2022-08-31 11:13:31

我有一个bean,我想使用Spring util命名空间注入一个命名列表,但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)

这让我感到困惑,因为我认为这将是预期的工作方式。


答案 1

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

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

这同样适用于类型化集合...

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

该解决方案在 3.11.3 中提到。使用限定符微调基于注释的自动布线

如果您打算按名称表示注释驱动的注入,则不要主要使用 - 即使在技术上能够通过值引用 Bean 名称。相反,更喜欢 JSR-250 注释,它在语义上被定义为通过其唯一名称来标识特定的目标组件,而声明的类型与匹配过程无关。@Autowired@Qualifier@Resource

作为这种语义差异的特定结果,本身被定义为集合或映射类型的bean不能通过@Autowired注入,因为类型匹配不适用于它们。用于此类豆子,通过唯一名称引用特定集合/映射豆。@Resource

因此,请在测试中使用它,并且它可以正常工作:

@Resource(name="myList") private List<String> stringList;

答案 2

可能发生的另一件事是,您正在自动连接Bean的属性。在这种情况下,您不需要自动连接它,而只需创建 setter 方法并在 bean 定义中使用属性标记(使用 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


推荐