3

我是 Java 和 Spring 的新手,来自 C# 和 .NET 世界,所以请耐心等待 - 我试图做的事情可能不合时宜......

我正在尝试使用 Java 配置和注释而不是 XML 配置来配置 Spring DI,但是我遇到了一些问题。这适用于独立应用程序,而不是 Web 应用程序。我已经阅读了 springsource 文档,据我所知,我的基本配置应该是正确的......但不是。请看下面的代码:

Java 配置注解类:

package birdalerter.common;

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;

import birdalerter.process.ISightingsProcessor;
import birdalerter.process.SightingsProcessor;

@Configuration
@ComponentScan({"birdalerter.process", "birdalerter.common"})
public class AppConfig {
    @Bean
    @Scope("prototype")
    public ISightingsProcessor sightingsProcessor(){
        return new SightingsProcessor();
    }
}

配置实现 ISightingsProcessor 接口的组件:

package birdalerter.process;

import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.LinkedBlockingQueue;

import org.springframework.stereotype.Component;

import birdalerter.domainobjects.IBirdSighting;
@Component
public class SightingsProcessor implements ISightingsProcessor{

    private LinkedBlockingQueue<IBirdSighting> queue;
    private List<ISightingVisitor> sightingVisitors = new ArrayList<ISightingVisitor>();

    public SightingsProcessor(){
    }

    ...
}

配置工厂组件:

package birdalerter.process;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Required;
import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Component;

@Component
public class ProcessorFactory {
    private ISightingsProcessor sightingsProcessor;

    @Autowired
    @Required
    private void setSightingsProcessor(ISightingsProcessor sightingsProcessor){
        this.sightingsProcessor = sightingsProcessor;
    }

    public ISightingsProcessor getSightingsProcessor(){
        return this.sightingsProcessor;
    }
}

连接 AnnotationConfigApplicationContext 并测试:

@Test
public void testProcessingDI(){
    @SuppressWarnings("resource")
    AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext();
    context.register(AppConfig.class);
    context.refresh();


    ISightingsProcessor processor = new ProcessorFactory().getSightingsProcessor();
    System.out.println(processor);
    Assert.assertTrue(processor != null);
}

SightingsProcessor 没有被设置器注入,并且断言失败,因为返回的对象为空。希望我错过了一些非常明显的东西。

提前致谢。

针对 Meriton 编辑:

谢谢美利通的回答。

为什么 Spring 不知道新创建的对象?Spring 是否不会在整个应用程序生命周期中维护依赖关系并在创建配置为 bean 的新对象时适当地注入?

老实说,如果我可以帮助它,我不想直接使用context.getBean(ISightingsProcessor.class)它,我希望在 setter 方法中注入依赖项而无需手动干预 - 它看起来更干净。

我正在使用接口扩展- 实现对象将作为线程启动ProcessorFactory。该应用程序将可配置为具有 n* 个线程,每个线程都在循环迭代中启动。我认为在方法声明中不可能有注释(我可能错了,请告知),因此我使用工厂来提供注入的具体类的新实例。ISightingsProcessorRunnable@AutowiredISightingsProcessor

是的,我刚刚查看了@Scope注释-您是对的,需要移至AppConfig @Bean声明(我在此编辑中已完成),谢谢。

4

1 回答 1

1
ISightingsProcessor processor = new ProcessorFactory().getSightingsProcessor();

这会调用 ProcessorFactory 的构造函数,然后调用构造函数创建的实例的 getter。Spring 无法知道该新创建的对象,因此不会注入其依赖项。您应该向 Spring 询问 ProcessorFactory,例如

ProcessorFactory pf = context.getBean(ProcessorFactory.class);
ISightingsProcessor processor = pf.getSightingsProcessor();

也就是说,我根本不知道你为什么需要类 ProcessorFactory。您也可以直接获取 ISightingsProcessor:

ISightingsProcessor processor = context.getBean(ISightingsProcessor.class);

此外,“基于 Java 的配置”和组件扫描是声明 bean 的独立方式。目前,您因此要声明 ISightingsProcessor 两次:一次使用带有 @Bean 注释的工厂方法,一次使用组件扫描和类上的 @Component 注释。做任何一个都可以。事实上,两者都做可能会导致一个 bean 定义覆盖另一个。

哦,@Scope注释是针对 bean 定义的(那些你用@Beanor注释的@Component)。它可能会在注入点 ( @Autowired) 上被忽略。

于 2013-01-20T13:47:36.210 回答