Spring并在运行时将参数传递给factory-method

时间:2011-07-22 02:05:09

标签: java spring factory-method

方法context.getBean(name,user)的文档说

  

允许指定显式构造函数参数/工厂方法   参数

但无论我做什么(尝试过所有内容),在最初的逻辑设置中,我在初始化期间加载bean时会得到这个:

org.springframework.beans.factory.UnsatisfiedDependencyException:
Error creating bean with name 'fileValidator' defined in
PortletContext resource
[/WEB-INF/classes/context/customer-form-portlet.xml]: Unsatisfied
dependency expressed through constructor argument with index 0 of type
[com.liferay.portal.model.User]: Ambiguous factory method argument
types - did you specify the correct bean references as factory method
arguments?
    org.springframework.beans.factory.UnsatisfiedDependencyException:
Error creating bean with name 'fileValidator' defined in
PortletContext resource
[/WEB-INF/classes/context/customer-form-portlet.xml]: Unsatisfied
dependency expressed through constructor argument with index 0 of type
[com.liferay.portal.model.User]: Ambiguous factory method argument
types - did you specify the correct bean references as factory method
arguments?

<bean id="fileValidator" 
      class="cz.instance.transl.validation.file.FileValidator" 
      factory-method="createInstance" />

private FileValidator(User user) {
    this.user = user;
}

public static FileValidator createInstance(User user) {
    return new FileValidator(user);
}

评论说你可以这样做,但如果你在那个bean的xml定义中指定构造函数参数,它就会失败。

4 个答案:

答案 0 :(得分:16)

javadoc说:

  

args - 使用静态工厂方法的显式参数创建原型时使用的参数。

因此bean定义必须是一个原型范围的bean,即

<bean id="fileValidator" 
      scope="prototype" 
      class="cz.instance.transl.validation.file.FileValidator" 
      factory-method="createInstance" />

答案 1 :(得分:10)

阅读了20篇文章,我发现如何获得自定义工厂方法在运行时获取参数并不明显,特别是因为我们被迫使用构造函数-arg标记并引用现有的bean上下文设置如下,有问题的类充当静态工厂方法。

<bean id="user" class="something.something.User" />

<bean id="fileValidator" 
      class="cz.instance.transl.validation.file.FileValidator" 
      factory-method="createInstance" >
      <constructor-args ref="user" />
</bean>

我通过从上下文中获取构造函数-arg中使用的bean实例,然后使用您在运行时使用的值填充它来实现它。当您获得工厂生成的bean时,此bean将用作参数。

public class X {

   public void callFactoryAndGetNewInstance() {
      User user = context.getBean("user");
      user.setSomethingUsefull(...);
      FileValidator validator = (FileValidator)context.getBean("fileValidator");
      ...
   }
}

请注意,这并不能解决使用context.getBean(arg1,arg2)时遇到的问题,因为该方法与此方案无关。它不是因为所有这些bean都是单例,并且此时不调用构造函数。如果您在单用户系统中工作,这不是问题,也不需要关心,因为无论如何您在上下文中只有1个用户bean!

但是,对于多用户系统,您需要确保每个真实用户都有一个唯一的User bean,并且您在工厂方法调用中使用了正确的User bean。

为了在多用户系统中执行此操作,您需要将bean类型更改为原型并且您应该创建表示工厂的FileValidator的bean(如果您计划将依赖项注入工厂)和另一个表示新实例的bean FileValidator。它们都属于同一类类型,但您必须为每个类别指定一个唯一的名称。见下文:

<bean id="user" scope="prototype" class="something.something.User" />

<bean id="validatorFactory"
            class="cz.instance.transl.validation.file.FileValidator">
    <constructor-arg value="something" />
</bean>

<bean id="fileValidatorBean"
            class="cz.instance.transl.validation.file.FileValidator"
    scope="prototype"
    factory-method="createInstance" >
    <constructor-arg ref="user" />
</bean>

并且在您希望从工厂获取此新FileValidator bean的类中,您可以使用以下技术:

public void someMethod() {
    ...
    User user = context.getBean("user");
    user.setSomethingUsefull(...);

    FileValidator fileValidator = 
               (FileValidator)context.getBean("fileValidatorBean",
                                              user);
    ...
}

答案 2 :(得分:7)

为了调用工厂方法,Spring需要访问用户实例以传递给createInstance。在这种情况下,我只是创建一个bean并将其传递给:

<bean id="user" class="something.something.User">
</bean>

<bean id="validator" class="cz.instance.transl.validation.file.FileValidator" factory-method="createInstance">
    <constructor-arg ref="user"/>
</bean>

答案 3 :(得分:6)

您也可以使用抽象工厂设置factory-bean属性。这里我们有一个ActionFactory来创建动作。

<bean id="actions_factory" class="com.imagina.control.actions.impl.ActionFactoryImpl"/>

<bean id="load_person_action" class="com.imagina.control.actions.impl.LoadPersonAction" 
  factory-bean="actions_factory" factory-method="create">
  <constructor-arg value="load_person_action"/>      
</bean>

要使用此配置,您必须考虑以下几点:

  1. create方法不是静态的。现在属于一个实例
  2. constructor-arg是工厂方法的参数