4

如果我使用@Bean 声明一个类,然后对该类进行组件扫描,spring 将通过调用它的构造函数并注入构造函数参数并注入任何标有@Inject 的字段来实例化该类。为简单起见,我们将此称为 spring 自动构建。

我不喜欢组件扫描并希望完全避免它(我不想讨论我不喜欢它的原因)。我想改用 @Configuration 对象,但仍然希望我可以使用自动构建功能。是否可以调用 spring 来自动构建我的对象,而不是显式地传递我的 @Configuration 对象中的所有构造函数参数?

假设我有一个 bean:

public class MyServiceImpl implements MyService {
    public MyServiceImpl(Dependency1 d1, Dependency d2) { ... }
    ....
}

我可以像这样定义一个配置对象:

@Configuration
public class MyConfiguration {
    // lets assume d1 and d2 are defined in another @Configuration
    @Inject
    Dependency1 d1; 

    @Inject
    Dependency2 d2;

    @Bean
    public MyService myService() { 
        // I dislike how I have to explicitly call the constructor here
        return new MyServiceImpl(d1, d2);
    }
}

但是现在,我必须自己明确地调用 MyServiceImpl 构造函数,所以我必须不断更新它,因为我的构造函数会随着时间的推移而发生变化。

我希望我可以声明一个抽象方法,以便可以进行春季自动构建:

@Configuration
public abstract class MyConfiguration {
    @Bean
    public abstract MyServiceImpl myService();
}

但这不起作用。有没有一种方法可以在不使用组件扫描的情况下调用 spring 自动构建?

在 Google Guice 中,这可以通过 Binder 完成: https ://google-guice.googlecode.com/svn/trunk/javadoc/com/google/inject/Binder.html

在 Tapestry IOC 中,这可以通过 ServiceBinder 完成: http: //tapestry.apache.org/ioc-cookbook-basic-services-and-injection.html#IoCCookbook-BasicServicesandInjection-SimpleServices

更新

根据 spod 的回答,我能够实现我所追求的(谢谢!)。为任何想要做同样事情的人提供测试用例:

import java.util.Date;
import javax.inject.Inject;
import junit.framework.Assert;
import org.junit.Test;
import org.springframework.beans.factory.config.AutowireCapableBeanFactory;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

public class AutoBuildConfigurationTest {
    @Configuration
    public static class MyConfiguration {
        @Inject
        private AutowireCapableBeanFactory beanFactory;

        @Bean
        public Date date() {
            return new Date(12345);
        }

        @Bean
        public MyService myService() {
            return autoBuild(MyService.class);
        }

        protected <T> T autoBuild(Class<T> type) {
            return type.cast(beanFactory.createBean(type, AutowireCapableBeanFactory.AUTOWIRE_CONSTRUCTOR, true));
        }
    }

    public static class MyService {
        private Date date;

        public MyService(Date date) {
            this.date = date;
        }

        public Date getDate() {
            return date;
        }
    }

    @Test
    public void testAutoBuild() {
        ApplicationContext appContext = new AnnotationConfigApplicationContext(MyConfiguration.class);
        MyService myService = appContext.getBean(MyService.class);
        Assert.assertEquals(12345, myService.getDate().getTime());
    }
}
4

1 回答 1

7

基于 java 的容器配置不依赖于以任何方式进行组件扫描。它只是基于 XML 的组件配置的不同方法。使用 XML 配置,您只需要使用 MyServiceImpl 类声明您的 bean,以防它已经被 @inject 注释。Spring 会识别注释并处理它们。如果您真的想在不自己调用构造函数的情况下从 @Configuration java 类中实例化 MyServiceImpl,那么您必须使用 bean 工厂(没有测试过,试一试):

@Configuration
public class MyConfiguration {

    @Autowired AutowireCapableBeanFactory beanFactory;

    @Bean public MyService myService() { 
        return beanFactory.createBean(MyServiceImpl.class, AutowireCapableBeanFactory.AUTOWIRE_CONSTRUCTOR, true);
    }
}
于 2012-11-27T12:30:05.410 回答