我可以为EJB使用CDI构造函数注入吗?

Are*_*rff 16 java ejb cdi

我想做这样的事情:

@Stateless
public class GreeterEjb {


    private final Greeter greeter;


    @Inject
    public GreeterEjb(Greeter greeter) {
        this.greeter = greeter;
    }


    public String greet() {
        return greeter.greet();
    }
}
Run Code Online (Sandbox Code Playgroud)

我用Glassfish 3.1.1和JBoss 7.0.2尝试了混合结果.在某些情况下,它可以工作,在其他情况下则不然.如果您对详细信息感兴趣,请参阅Glassfisch论坛中的此主题.

EJB 3.1规范,部分4.9.2 bean类说:

该类必须具有不带参数的公共构造函数.

这听起来像EJB不允许构造函数注入.

CDI规范在第3节开头说,会话Bean受CDI支持.然后,第3.2节详细讨论了CDI和EJB,但从未提及有关构造函数注入不起作用的任何内容.这让我觉得它应该被允许.

那么,这些规范是否允许为EJB提供CDI构造函数注入?

Are*_*rff 20

KrisPete Muir最终说服了我:即使使用另一个构造函数进行注入,EJB也必须有一个公共的无参数构造函数.很奇怪,同时使用两个构造函数,但它的工作原理.多谢你们.

在Glassfish 3.1.1,JBoss 7.0.2和TomEE 1.0.0-beta-2上成功测试.

@Stateless
public class GreeterEjb {

    private final Greeter greeter;


    @Inject
    public GreeterEjb(Greeter greeter) {
        this.greeter = greeter;
    }


    // public no-arg constructor required for EJBs
    // injection still works fine with the @Inject constructor
    public GreeterEjb() {
        this.greeter = null;
    }


    public String greet() {
        return greeter.greet();
    }
}
Run Code Online (Sandbox Code Playgroud)

  • 是的,我还考虑过在构造函数中抛出一个异常.事实证明,实际上调用了构造函数并抛出异常会阻止实例化EJB. (8认同)
  • 小技巧可能是让无参数构造函数抛出 IllegalStateException,因为它并不打算实际使用。如果您有 @Inject 构造函数,我认为 TomEE 允许省略无参数构造函数。如果没有,它很快就会:) (2认同)

小智 9

仅当为jar启用CDI时,Java EE 6中才需要构造函数注入EJB.如果这不适用于appserver,请提交bug.

还请在此处提交一个问题 - http://java.net/jira/browse/EJB_SPEC - 修复EJB语言规范(这是错误的).

这是在CDITCK中测试的 - https://github.com/jboss/cdi-tck/blob/master/impl/src/main/java/org/jboss/cdi/tck/tests/implementation/enterprise/definition/ExplicitConstructorSessionBean .java - 但不适用于no-interface-views,所以请在https://issues.jboss.org/browse/CDITCK中提出一个问题,我们可以为你的案例添加一个测试.