将参数传递给@Inject Bean的实例

时间:2014-04-10 17:15:16

标签: java cdi

我使用CDI作为注入框架,但我发现它的用法存在一些限制,这就是其中之一。我正在尝试使用运行时值初始化bean实例的创建。例如:

@RequestScoped
public class MyNumber {
   int number;

   public MyNumber(int number) {
      this.number = number;
   }

   public String toString() {
      return "Your number is: " + number;
   }
}

public class UseNumber {
   @Inject
   Instance<MyNumber> number;

   public void doStuff() {
      int a = 8;
      MyNumber mN = number.select(a).get(); // ?? - Obviously this does not work.

      System.out.print(mN); // Should print: "Your number is: 8"
   }
}

请注意,“a”是示例中的常量,但实际上它是一个变量;我澄清了这一点,因此您不会使用@Producer发布答案,然后在MyNumber的构造函数中注入值。

现在,任何人都知道我该怎么做?

2 个答案:

答案 0 :(得分:11)

我不确定你要做什么,但据我所知,你想用注入点注释中的数据初始化bean,或者在运行时通过编程查找来初始化bean。您可以通过在bean中使用InjectionPoint元数据来实现此目的(唯一的约束是将bean放在依赖范围内)

你可以这样做。

首先创建一个具有非约束值的限定符。

@Qualifier
@Target({TYPE, METHOD, PARAMETER, FIELD})
@Retention(RUNTIME)
@Documented
public @interface Initialized {

    @Nonbinding int value() default 0; // int value will be store here 
}

您必须在bean上添加此限定符并在创建时分析InjectionPoint

@Initialized
public class MyNumber {
   int number;

   private int extractValue(InjectionPoint ip) {
    for (Annotation annotation : ip.getQualifiers()) {
        if (annotation.annotationType().equals(Initialized.class))
            return ((Initialized) annotation).value();
    }
    throw new IllegalStateException("No @Initialized on InjectionPoint");
  }

   @Inject
   public MyNumber(InjectionPoint ip) {
      this.number = extractValue(ip);
   }

   public String toString() {
      return "Your number is: " + number;
   }
}

您现在可以注入一个初始化的数字:

@Inject
@Initialized(8)
MyNumber number;

如果要在运行时决定初始化值,则必须使用编程查找:

首先为`@ Initialized``

创建注释文字
public class InitializedLiteral extends AnnotationLiteral<Initialized> implements Initialized {

    private int value;

    public InitializedLiteral(int value) {
        this.value = value;
    }

    @Override
    public int value() {
        return value;
    }
}

然后你可以使用Instance创建你的bean。

public class ConsumingBean {

    @Inject
    @Any
    Instance<MyNumber> myNumberInstance;

    public MyNumber getMyNumberBeanFor(int value) {
     return myNumberInstance.select(new InitializedLiteral(value)).get();
    }
    ...
}

请记住,仅当MyNumber处于依赖范围内时才有效,因为这是在每次注入时更改初始化值的唯一方法。

答案 1 :(得分:4)

根据规范,没有办法让bean有一个不简单的“不注入”构造函数( 3.1. Managed beans 3.9. Bean constructors )。

因此,设置参数的方法是为它们设置setMethod(),将它们作为bean中的@Inject字段或使用注释@Inject注释构造函数,并使构造函数的参数可注入( 5.5.1. Injection using the bean constructor

我希望,我回答了这个问题。