0

最近我学到了很多关于 EJB3.1 的知识,但显然还不够。这是我的悲惨故事...

我与可以上传批处理文件的 servlet 发生争执。servlet 验证文件,将文件数据存储在数据库中,并向队列发送已接收到新批次的消息。这一切都很好。

我有一个部署了 ejb-jar 的耳朵,它有一个 MDB,用于侦听新的批次接收到的消息。

它还有一个执行实际批处理的有状态 EJB(使用无接口视图)。EJB 具有对无状态 JPA 实体服务 bean 的引用,因此需要进行容器管理以获取容器注入的 EntityManager。

每当 MDB 接收到消息时,我都需要创建有状态处理器 bean 的实例。我找不到任何涵盖这种情况的示例/教程。

我曾尝试将有状态 bean 注入 MDB,但它不起作用:

[#|2011-11-24T13:25:45.470-0700|SEVERE|glassfish3.1.1|javax.enterprise.system.container.ejb.mdb.com.sun.ejb.containers|_ThreadID=21;_ThreadName=Thread-2;|MDB00050: Message-driven bean [MyProcessor-ear-1.0:MyMDB]: Exception in creating message-driven ejb : [com.sun.enterprise.container.common.spi.util.InjectionException: Exception attempting to inject Remote ejb-ref name=my.package.MyMDB/myEJB,Remote 3.x interface =my.package.MyEJB,ejb-link=null,lookup=,mappedName=,jndi-name=my.package.MyEJB,refType=Session into class my.package.MyMDB: Lookup failed for 'java:comp/env/my.package.MyMDB/myEJB' in SerialContext[myEnv={java.naming.factory.initial=com.sun.enterprise.naming.impl.SerialInitContextFactory, java.naming.factory.state=com.sun.corba.ee.impl.presentation.rmi.JNDIStateFactoryImpl, java.naming.factory.url.pkgs=com.sun.enterprise.naming}]|#]

显然这是一种糟糕的设计方法,因为 MDB 本质上是无状态的,因此不应该注入有状态的 EJB。这是有道理的。

我的问题是当 MDB 接收到消息并确保 EJB 是容器管理时,我还能如何实例化我的有状态 EJB?

4

3 回答 3

0

如果您需要像上面指出的那样独立于批处理维护单个状态(运行、暂停、取消),即该状态适用于在特定时刻消耗的所有 MDB:为什么不将 EJB 实现为@Singleton? 通过这种方式,您可以在 EJB 中的实例变量中拥有该状态,并从中读取 MDB。

于 2011-11-25T14:16:51.897 回答
0

我们重新设计了这些类并将它们合并,以便 MDB 的 onMessage 方法进行处理。个体批次的状态存储在 MDB 上的静态地图中。

于 2011-12-01T18:21:50.393 回答
0

您可以使用 javax.jms.QueueBrowser 来检查消息在队列中的状态,您可以使用选择器和属性消息来定位具体的批处理过程。

于 2011-12-10T11:11:41.353 回答