迁移到CDI 1.2 WELD异常

时间:2018-06-28 15:42:12

标签: java-ee cdi

在Liberty中迁移到CDI 1.2之后,我的应用程序开始引发WELD异常,我不清楚如何解决:

org.jboss.weld.exceptions.UnsatisfiedResolutionException: WELD-001334: Unsatisfied dependencies for type DataHandler with qualifiers @Any @DataHandlerBean WELD-001475: The following beans match by type, but none have matching qualifiers:
  - Session bean [class com.ibm.blueargus.feed.PathSquadHandler with qualifiers [@DataHandlerBean @Any]; local interfaces are [DataHandler],
  - Session bean [class com.ibm.blueargus.feed.DownAccessPointDataHandler with qualifiers [@DataHandlerBean @Named @Any]; local interfaces are [DataHandler],
  - Session bean [class com.ibm.blueargus.feed.ATTTicketDataHandler with qualifiers [@DataHandlerBean @Any]; local interfaces are [DataHandler],
  - Session bean [class com.ibm.blueargus.feed.WirelessClientDataHandler with qualifiers [@Named @DataHandlerBean @Any]; local interfaces are [DataHandler]

这种情况发生在我要基于自定义注释值选择bean的情况下,这是应用程序能够基于数据库中存储的标准执行特定EJB实现的要求。如何更新我的代码以与CDI 1.1一起使用,并仍然允许基于自定义限定符值动态选择EJB实现?有没有更好的方法可以赋予我相同的一般功能?

预选赛规范

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

   String value();

}

注释实现:

public class DataHandlerBeanQualifier extends AnnotationLiteral <DataHandlerBean> implements DataHandlerBean {

    private String value;

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

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

}

EJB接口:

@Local
public interface DataHandler {

    public void extractData (Monitor monitor, DataFeed dataFeed) throws Exception;

}

示例EJB实现:

@Stateless
@DataHandlerBean("downAccessPointHandler")
public class DownAccessPointDataHandler implements DataHandler {

引起问题的业务逻辑EJB(摘要): monitor.getDataHandler()中的值用于选择适当的EJB实现以将控制权传递给进一步处理。

@Stateless
public class DataScannerImpl implements DataScanner {

    public static Logger logger = Logger.getLogger(DataScanner.class.getName());

    @Inject
    @Any
    private Instance <DataHandler> dataHandlerInstance;

    private String processMonitorList(List <Monitor> monitorList, boolean forceLoad) {
        ...

        try {
            dataHandler = dataHandlerInstance.select(new DataHandlerBeanQualifier(monitor.getDataHandler())).get();
            dataHandler.extractData(monitor, dataFeed);
            monitor.setDataFeed(dataFeed);
        } catch ( Exception e ) {
            e.printStackTrace();
            logger.warn("Could not execute dataHandler: " + monitor.getDataHandler(), e);
        }

        ...
    }
}

在我以前的WebSphere v8.5生产环境中,一切都很好。该Web应用程序使用的是CDI 1.0。

当前环境(server.xml):

<featureManager>
    <feature>javaee-7.0</feature>
    <feature>webProfile-7.0</feature>
    <feature>ejbPersistentTimer-3.2</feature>
    <feature>ldapRegistry-3.0</feature>
    <feature>appSecurity-2.0</feature>
    <feature>localConnector-1.0</feature>
    <feature>jaxrs-2.0</feature>
    <feature>jaxrsClient-2.0</feature>
    <feature>servlet-3.1</feature>
    <feature>jpa-2.1</feature>
</featureManager>

2 个答案:

答案 0 :(得分:0)

您必须具有一个@Producer方法,该方法将与实现DataScanner接口的4个类一起注入
必须为4个类添加4个不同的注释
根据您的业务逻辑,生产者将选择并“生产”正确的实施方案
然后,您可以在需要的地方直接注入"DataScanner"

@Inject
private DataHandler dataHandlerInstance;

这在Weld documentatiion here

中有描述

在文档中,生产者在"PaymentProcessor"接口的两种实现之间进行选择:"@Synchronous PaymentProcessor syncPaymentProcessor""@Asynchronous PaymentProcessor asyncPaymentProcessor"

答案 1 :(得分:0)

问题是由数据库值中的错字引起的,该错字导致monitor.getDataHandler()的值返回实际上不存在的bean名称。