gpt4 book ai didi

java - 使用 Java 容器配置的 Spring 依赖注入(inject)

转载 作者:塔克拉玛干 更新时间:2023-11-02 20:10:25 25 4
gpt4 key购买 nike

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

我正在尝试使用 Java 配置和注释而不是 XML 配置来配置 Spring DI,但是我遇到了一些问题。这是一个独立的应用程序,而不是一个网络应用程序。我已经完成了 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 接口(interface)的组件:

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 没有被 setter 注入(inject),断言失败,因为返回的对象为空。希望我错过了一些非常明显的东西。

提前致谢。

为回应 Meriton 而编辑:

感谢 Meriton 的回答。

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

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

我正在使用 ProcessorFactory作为ISightingsProcessor接口(interface)扩展 Runnable - 执行对象将作为线程启动。该应用程序将可配置为具有 n* 个线程,每个线程都在循环迭代中启动。我认为不可能(我可能错了,如果是的话请指教)@Autowired方法声明中的注释,因此我使用工厂提供注入(inject)的新实例 ISightingsProcessor具体类。

是的,我刚刚查看了 @Scope注释 - 你是对的,需要移动到 AppConfig @Bean声明(我在这次编辑中所做的),谢谢。

最佳答案

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

这会调用 ProcessorFactory 的构造函数,然后调用构造函数创建的实例的 getter。 Spring 不知道那个新创建的对象,因此不会注入(inject)它的依赖项。您应该向 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 定义(那些用 @Bean@Component 注释的)。它可能会在注入(inject)点 (@Autowired) 上被忽略。

关于java - 使用 Java 容器配置的 Spring 依赖注入(inject),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14424930/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com