猿问

Spring Java Config:如何使用运行时参数创建原型范围的@Bean?

使用Spring的Java Config,我需要使用只能在运行时获得的构造函数参数来获取/实例化作用域原型的bean。考虑以下代码示例(为简便起见,对其进行了简化):


@Autowired

private ApplicationContext appCtx;


public void onRequest(Request request) {

    //request is already validated

    String name = request.getParameter("name");

    Thing thing = appCtx.getBean(Thing.class, name);


    //System.out.println(thing.getName()); //prints name

}

Thing类的定义如下:


public class Thing {


    private final String name;


    @Autowired

    private SomeComponent someComponent;


    @Autowired

    private AnotherComponent anotherComponent;


    public Thing(String name) {

        this.name = name;

    }


    public String getName() {

        return this.name;

    }

}

注意事项name是final:它只能通过构造函数来提供,并保证不变性。其他依赖关系是Thing类的特定于实现的依赖关系,不应知道(紧密耦合到)请求处理程序实现。


这段代码与Spring XML配置完美配合,例如:


<bean id="thing", class="com.whatever.Thing" scope="prototype">

    <!-- other post-instantiation properties omitted -->

</bean>

如何使用Java配置实现同一目的?以下内容在Spring 3.x中不起作用:


@Bean

@Scope("prototype")

public Thing thing(String name) {

    return new Thing(name);

}

现在,我可以创建一个工厂,例如:


public interface ThingFactory {

    public Thing createThing(String name);

}

但这打败了使用Spring替换ServiceLocator和Factory设计模式的全部观点,这对于该用例而言是理想的选择。


如果Spring Java Config可以做到这一点,我将能够避免:


定义工厂接口

定义工厂实现

为工厂实施编写测试

相对于琐碎的事情,Spring已经通过XML配置支持了很多工作(相对而言)。


白衣非少年
浏览 818回答 3
3回答

哈士奇WWW

在@Configuration课堂上,@Bean像这样的方法@Bean@Scope("prototype")public Thing thing(String name) {&nbsp; &nbsp; return new Thing(name);}用于注册bean定义并提供创建bean的工厂。它定义的Bean仅在请求时使用直接或通过扫描确定的参数实例化ApplicationContext。对于prototypeBean,每次都会创建一个新对象,因此@Bean也会执行相应的方法。您可以ApplicationContext通过的BeanFactory#getBean(String name, Object... args)方法检索Bean,该方法指出允许指定显式构造函数自变量/工厂方法自变量,覆盖Bean定义中指定的默认自变量(如果有)。参数:如果使用静态工厂方法的显式参数创建原型,则使用args参数。在任何其他情况下,使用非null的args值都是无效的。换句话说,对于此prototype作用域的bean,您将提供将要使用的参数,而不是在bean类的构造函数中,而是在@Bean方法调用中。对于Spring 4+版本至少是这样。

慕标琳琳

使用Spring> 4.0和Java 8,您可以更安全地执行此操作:@Configuration&nbsp; &nbsp;&nbsp;public class ServiceConfig {&nbsp; &nbsp; @Bean&nbsp; &nbsp; public Function<String, Thing> thingFactory() {&nbsp; &nbsp; &nbsp; &nbsp; return name -> thing(name); // or this::thing&nbsp; &nbsp; }&nbsp;&nbsp; &nbsp; @Bean&nbsp; &nbsp; @Scope(value = "prototype")&nbsp; &nbsp; public Thing thing(String name) {&nbsp; &nbsp; &nbsp; &nbsp;return new Thing(name);&nbsp; &nbsp; }}用法:@Autowiredprivate Function<String, Thing> thingFactory;public void onRequest(Request request) {&nbsp; &nbsp; //request is already validated&nbsp; &nbsp; String name = request.getParameter("name");&nbsp; &nbsp; Thing thing = thingFactory.apply(name);&nbsp; &nbsp; // ...}因此,现在您可以在运行时获取bean。当然,这是一种工厂模式,但是您可以节省一些时间来编写特定的类,例如ThingFactory(但是您必须编写custom @FunctionalInterface来传递两个以上的参数)。
随时随地看视频慕课网APP
我要回答