CDI:注入单个实例有效,但注入 Instance<> 无效。为什么?

问题描述 投票:0回答:1

我正在尝试借助 CDI 注入来实现某种“插件功能”。但我遇到了一些麻烦,我想我可能会从我的“Stackers”同伴那里得到一些外部观点:-)

到目前为止,我已经做到了这样的事情:

@Inject 
@ScoringModule("AGE")
private AgeModule ageModule;

@Inject 
@ScoringModule("CUSTOMER_TYPE")
private CustomerTypeModule customerTypeModule;

当我“运行”这个时,两个字段都会注入适当的实例。但是当我尝试像这样更“动态”地注入它们时:

@Inject @Any
private Instance<CustomerScoringModule> scoringModuleInstance;

CustomerScoringModule module = scoringModuleInstance.select(new ScoringModuleLiteral("AGE")).get();

然后我会得到这样的异常:

org.jboss.weld.exceptions.UnsatisfiedResolutionException: WELD-001334: Unsatisfied dependencies for type CustomerScoringModule with qualifiers @Any @ScoringModule

这对我来说似乎很奇怪,因为 CDI 显然“知道”我感兴趣的实例,因为直接注入到类型字段中是有效的。所以我认为这一定与预选赛有关?不知何故,CDI“看到”了这两个实例,但认为它们都不适合我的注入请求?难道是这样吗?

有什么“明显”的事情浮现在脑海中,我可能在这里做错了吗?我第一次尝试使用 CDI,有可能(甚至很可能)我在某个地方做了一些愚蠢的事情:-)

感谢任何帮助或提示,并提前非常感谢!

下面我将附加限定符的“周围”类和注释文字等以供参考。

两个模块都实现的 CustomerScoringInterface:

package de.otto.cccs.customerscoring.modules;

import de.otto.cccs.customerscoring.modules.vo.ScoringModuleResponseVO;
import de.otto.cccs.customerscoring.valueobjects.webservice.ScoringRequestVO;

public interface CustomerScoringModule {
    public ScoringModuleResponseVO process(ScoringRequestVO inputVO);
}

两个模块实现之一(它们现在只是虚拟实现,唯一的区别是 process() 方法中的 log.info() 输出,这就是为什么我在这里只包含其中之一):

package de.otto.cccs.customerscoring.modules;

import javax.ejb.LocalBean;
import javax.ejb.Stateless;

import de.otto.cccs.customerscoring.modules.qualifiers.ScoringModule;
import de.otto.cccs.customerscoring.modules.vo.ScoringModuleResponseVO;
import de.otto.cccs.customerscoring.valueobjects.webservice.ScoringRequestVO;
import lombok.extern.log4j.Log4j2;

@Log4j2
@Stateless
@LocalBean
@ScoringModule("AGE")
public class AgeModule implements CustomerScoringModule {

    public AgeModule() {
        super();
    }

    @Override
    public ScoringModuleResponseVO process(ScoringRequestVO inputVO) {
        log.info("processed AGE_MODULE");

        return new ScoringModuleResponseVO();
    }

}

@ScoringModule 限定符:

package de.otto.cccs.customerscoring.modules.qualifiers;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

import javax.inject.Qualifier;

@Qualifier
@Target({ElementType.TYPE, ElementType.METHOD, ElementType.FIELD, ElementType.PARAMETER})
@Retention(RetentionPolicy.RUNTIME)
public @interface ScoringModule {
    String value();
}

以及注释文字:

package de.otto.cccs.customerscoring.modules.qualifiers;

import javax.enterprise.util.AnnotationLiteral;

public class ScoringModuleLiteral extends AnnotationLiteral<ScoringModule> implements ScoringModule {

    private static final long serialVersionUID = 1L;
    private String value;

    public ScoringModuleLiteral(String value) {
        this.value = value;
    }

    @Override
    public String value() {
        return this.value;
    }
}

最后是我的空 beans.xml:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/beans_1_0.xsd">
</beans>
java jakarta-ee cdi payara
1个回答
0
投票

我最终将模块从 EJB 更改为普通 Java 类,所有麻烦都消失了。

我并不真正需要 EJB 的模块,我只是在开始时错误地认为它们必须通过 CDI 注入,但现在情况不再是这样了(对于旧的 J2EE 版本来说确实如此)

感谢您的反馈!

© www.soinside.com 2019 - 2024. All rights reserved.