2

我试图从另一个 SO 问题中实施解决方案: https ://stackoverflow.com/a/10059245/1943765但它并非在所有情况下都有效。

当我将此代码放入 CDI bean 时(bean 代码显示在底部):

@Inject @HttpParam
private String code; 

一切正常。但是当我尝试为 @HttpParam 注释定义值时(所以,不是默认值) Weld 无法启动:

@Inject @HttpParam(value="code")
private String token;

我得到这个例外:

Caused by: org.jboss.weld.exceptions.DeploymentException: WELD-001408: Unsatisfied dependencies for type String with qualifiers @HttpParam
  at injection point [BackedAnnotatedField] @Inject @HttpParam private org.test.site.jsf.beans.request.ActivationBean.token
  at org.test.site.jsf.beans.request.ActivationBean.token(ActivationBean.java:0)
WELD-001475: The following beans match by type, but none have matching qualifiers:
  - Producer Method [String] with qualifiers [@HttpParam @Any] declared as [[BackedAnnotatedMethod] @Produces @HttpParam public org.test.site.cdi.producer.HttpParamProducer.getHttpParameter(InjectionPoint)]
at org.jboss.weld.bootstrap.Validator.validateInjectionPointForDeploymentProblems(Validator.java:359)
    at org.jboss.weld.bootstrap.Validator.validateInjectionPoint(Validator.java:281)
    at org.jboss.weld.bootstrap.Validator.validateGeneralBean(Validator.java:134)
    at org.jboss.weld.bootstrap.Validator.validateRIBean(Validator.java:155)
    at org.jboss.weld.bootstrap.Validator.validateBean(Validator.java:518)
    at org.jboss.weld.bootstrap.ConcurrentValidator$1.doWork(ConcurrentValidator.java:68)
    at org.jboss.weld.bootstrap.ConcurrentValidator$1.doWork(ConcurrentValidator.java:66)
    at org.jboss.weld.executor.IterativeWorkerTaskFactory$1.call(IterativeWorkerTaskFactory.java:63)
    at org.jboss.weld.executor.IterativeWorkerTaskFactory$1.call(IterativeWorkerTaskFactory.java:56)
    ... 4 more

我使用的代码类似于链接的 SO 问题。

自定义 @HttpParam 注释:

@Qualifier
@Retention(RUNTIME)
@Target({METHOD, FIELD, PARAMETER})
public @interface HttpParam {
    public String value() default "";
}

注释值生产者:

public class HttpParamProducer {

    @Inject
    FacesContext facesContext;

    @Produces
    @HttpParam
    String getHttpParameter(InjectionPoint ip) {
        String name = ip.getAnnotated().getAnnotation(HttpParam.class).value();
        if ("".equals(name)) name = ip.getMember().getName();
        return facesContext.getExternalContext()
                .getRequestParameterMap()
                .get(name);
    }
}

使用它的 CDI bean 类似于:

@Named("activation")
@RequestScoped
public class ActivationBean implements Serializable{

    @Inject
    @HttpParam(value="code")
    private String token;

    // rest of valid class code
}

我也在使用带有 Weld Servlet 2.3.1.Final 的 Tomcat 8 服务器。

所以..我做错了什么?:-/

4

3 回答 3

3

您需要添加@Nonbinding到您的value属性,以便它的值不一定选择调用哪个生产者方法,但允许单个工厂样式生产者检查您的注入点。

您可能还想查看应该在 Java EE7 中的限定符中 @Nonbinding 注释的目的是什么?

于 2015-12-12T06:52:39.123 回答
3

默认情况下,CDI Qualifier 中的属性也用于确定要注入的 bean。

既然你注入@HttpParam(value="code"),但你的生产者只是生产 @HttpParam(value="")。它们不匹配,CDI 不知道要注入什么。

要使 CDI 在确定要注入哪个 bean 时忽略 CDI Qualifier 中的属性,可以@Nonbinding在这些属性上进行标记

@Qualifier
@Retention(RUNTIME)
@Target({METHOD, FIELD, PARAMETER})
public @interface HttpParam {
    @Nonbinding
    public String value() default "";
}
于 2015-12-12T07:01:41.363 回答
0

我迟到了,但为了其他可能使用这个问题来编写类似的基于参数的注入的人,让我添加这个。

添加一个我在上面没有找到的具体内容。我遇到了同样的情况,但风格有所不同。我Qualifier的方法和Producer方法位于一个模块中(比如injector-module),与我进行注射的模块(比如)不同injected-module。给出了所有NonBinding注释。

但是,我只在injected-module.

注射失败。

在 META-INF 中添加 beans.xmlinjector-module就可以了。

于 2021-05-23T16:09:23.377 回答