我是Java和Spring的新手,来自C#和.NET世界,所以请耐心等待 - 我试图做的事情可能不合时宜......
我正在尝试使用Java配置和注释配置Spring DI,而不是XML配置,但是我遇到了一些问题。这适用于独立应用程序,而不是Web应用程序。我已经完成了the springsource documentation,据我所知,我的基本配置应该是正确的......但事实并非如此。请看下面的代码:
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未被设置者注入,并且断言失败,因为返回的对象为null。希望我错过了一些非常明显的东西。
提前致谢。
在对Meriton的回应中编辑:
感谢Meriton的回答。
为什么Spring不知道新创建的对象? Spring是否在整个应用程序生命周期中不保持依赖关系,并在创建配置为bean的新对象时适当地注入?
我不想直接使用context.getBean(ISightingsProcessor.class)
如果我可以帮助它说实话,我希望在没有人工干预的情况下在setter方法中注入依赖 - 它看起来更干净。
我使用ProcessorFactory
作为ISightingsProcessor
接口扩展Runnable
- 实现对象将作为线程启动。应用程序可配置为具有n *个线程,每个线程在循环迭代中启动。我不认为有可能(我可能错了,请告知是否这样)在方法声明中有@Autowired
注释,因此我使用工厂提供注入的ISightingsProcessor
具体的新实例类。
是的,我刚看了@Scope
注释 - 你是对的,需要转到AppConfig
@Bean
声明(我在此编辑中已完成) ,谢谢你。
答案 0 :(得分:0)
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-annotated工厂方法,一次使用组件扫描和类上的@Component注释。做其中任何一件都行。实际上,两者都可能导致一个bean定义覆盖另一个。
哦,@Scope
注释用于bean定义(使用@Bean
或@Component
注释的那些)。注入点(@Autowired
)可能会被忽略。