0

我有 2 个应用程序。一个“主”,另一个带有远程 EJB。第一个应用程序在第二个应用程序中调用远程 EJB。

现在我想实现 Observable 模式。但它似乎不适用于跨应用程序。

所以我希望我的 Observable 在第二个应用程序中,而我的 Observers 在第一个应用程序中。当我尝试它时,它似乎不起作用。只有当我在同一个应用程序中创建 Observable 和 Observers 时它才有效。

我有另一个解决方案,但我认为它很脏。我可以在第二个应用程序中使用 MDB,并在主题上发布一条消息,然后在第一个应用程序中使用这些消息。但话又说回来,这似乎有点脏。

我在 Payara 4.1.1.171 服务器上使用 JEE7。

编辑:因为我发现触发的事件不会退出 EAR 文件,所以我现在使用 MDB 方法。但这仍然行不通。更多信息,这是 EAR 文件的包结构。

my-application.ear
+--- common-ejb.jar (EJB)
|       +--- MessageDrivenBean.java (Triggers the event)
+--- my-application.jar (EJB)
+--- my-application.war (WEB)
        +--- WEB-INF
                +--- lib
                |       +--- common-web.jar (Resources WEB jar)
                |           +--- SessionBean1 to receive the events
                +--- classes
                        +--- SessionBean2 to receive the events

如您所见,有 2 个会话 bean 来接收事件,但 2 个实际上都没有接收到它们。

4

2 回答 2

0

启用 Payara Server 171 版和 Hazelcast 后,您可以触发 CDI 事件并在另一个应用程序中观察它们。请参阅有关远程 CDI 事件的文档

如果两个应用程序都部署在同一个 Payara Server 实例上,您将触发 MyEvent.class 类型的事件,如下所示:

@Inject
@Outbound(loopBack = true) // loopBack true is required if the observing application is deployed on the same Payara Server instance
Event<MyEvent> event;

...
event.fire(new MyEvent());

另一个应用程序将通过以下方式观察这些事件:

public void receiveEvent(@Observes @Inbound MyEvent event) {
}

如果观察应用程序在同一个 Hazelcast 集群中的另一个 Payara 服务器实例上运行,它也会收到该事件。在这种情况下,不需要 loopBack 属性,但也不会造成伤害。

远程 CDI 事件通过嵌入式 Hazelcast 数据网格进行调度,该数据网格将可靠地将事件传递给集群中的所有观察应用程序。观察应用程序必须在事件触发时运行 - 类似于通过 JMS 主题广播事件的行为。

于 2017-03-31T08:10:48.840 回答
0

给定一个消息驱动的 bean:

@MessageDriven(activationConfig = {
        @ActivationConfigProperty(propertyName = "destinationType", propertyValue = "javax.jms.Queue"),
        @ActivationConfigProperty(propertyName = "destinationLookup", propertyValue = "java:jboss/queue/SampleQueue")
})
public class EventListener implements MessageListener {

    @Inject
    @Any
    Event<MyEvent> loggedInEvent;

    @Override
    public void onMessage(Message message) {
        if (message instanceof ObjectMessage) {
            try {
                ObjectMessage objectMessage = (ObjectMessage) message;
                loggedInEvent.fire(((MyEvent) objectMessage.getObject()));
            } catch (JMSException e) {
                e.printStackTrace();
            }
        }
    }
}

其中 MyEvent 是一个简单的 POJO:

public class MyEvent implements Serializable {

    private final String payload; 

    public MyEvent(String payload) {
        this.payload = payload;
    }

    public String getPayload() {
        return payload;
    }
}

然后这个@SessionScoped bean:

@SessionScoped
public class EventHandler {

    public void handleEvent(@Observes MyEvent myEvent) {
        System.out.println(myEvent.getPayload());
    }
}

当一个MyEvent对象被发送到与EventListener

可以提供任意数量的观察者。

于 2017-03-30T14:03:04.717 回答