CDI事件已触发,但并非所有线程/会话都收到

时间:2018-06-25 16:12:50

标签: java cdi

我有一个简单的设置,其中只有一个Session支持JSF xhtml文件。在该事件中,我触发了一个事件,表明在同一会话和任何其他会话中,提交时都会收到该事件。

但是,很奇怪的是,我可以看到在事件触发期间,只有当前会话接收到它,而没有其他会话接收到它。我通过使用两种不同的浏览器(在这种情况下为Safari和Firefox)来确保有两个不同的会话。

我对基于CDI的事件的概念有误吗?

支持会话的Bean:

package testevent;

import java.io.Serializable;
import javax.enterprise.context.SessionScoped;
import javax.enterprise.event.Event;
import javax.enterprise.event.Observes;
import javax.enterprise.event.Reception;
import javax.inject.Inject;
import javax.inject.Named;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;

@Named
@SessionScoped
public class TestEventSession implements Serializable {
  private String message = "Start Message";
  private String receivedMessage = "";

  @Inject
  @ForTest Event<String> messageEvent;

  Logger LOG = LogManager.getLogger();

  public void messageChanged(@Observes(notifyObserver = Reception.IF_EXISTS) @ForTest String message) {
    LOG.info("messageChanged <-- "+message);
    this.receivedMessage = message;
  }

  public String getReceivedMessage() {
    return receivedMessage;
  }

  public String getMessage() {
    LOG.info("getMessage --> "+message);
    return message;
  }

  public void setMessage(String message) {
    LOG.info("setMessage <-- "+message);
    this.message = message;
    LOG.info("Firing Message Change");
    messageEvent.fire(message);
    LOG.info("Done Firing Message Change");
  }
}

xhtml文件:

<?xml version='1.0' encoding='UTF-8' ?> 
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html 
  xmlns="http://www.w3.org/1999/xhtml"
  xmlns:h="http://xmlns.jcp.org/jsf/html"
  xmlns:f="http://java.sun.com/jsf/core"
  >
  <f:view transient="false">
    <h:body>
      <h:form>
        <h:inputText value="#{testEventSession.message}" />
        <h:outputText value="#{testEventSession.receivedMessage}" />
        <h:commandButton value="Submit"/>
        <h:button value="Refresh" />
      </h:form>
    </h:body>
  </f:view>
</html>

使用的限定词是一个非常基本的限定词:

package testevent;

import static java.lang.annotation.ElementType.FIELD;
import static java.lang.annotation.ElementType.METHOD;
import static java.lang.annotation.ElementType.PARAMETER;
import static java.lang.annotation.ElementType.TYPE;
import static java.lang.annotation.RetentionPolicy.RUNTIME;

import java.lang.annotation.Retention;
import java.lang.annotation.Target;
import javax.inject.Qualifier;

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

作为调试输出的一部分,您可以看到只有当前线程/会话中的观察者方法接收到该事件:

10:45:11.213 [GUEST] [http-nio-8080-exec-32] INFO  testevent.TestEventSession - Firing Message Change
10:45:11.213 [GUEST] [http-nio-8080-exec-32] INFO  testevent.TestEventSession - messageChanged <-- Start Message #3
10:45:11.213 [GUEST] [http-nio-8080-exec-32] INFO  testevent.TestEventSession - Done Firing Message Change

请注意,这已经在TomEE 7.0.3中进行了测试,该软件假定使用的是OpenWebBeans 1.7.3。

相关

更新

这可能是预期的功能,但是我找不到CDI-Spec中明确说明的功能。尽管仅当Observer仅观察同一会话内的触发事件时,提供的示例才有意义。对于Produces,它清楚地说明了与作用域bean的互通。我认为这是一些官方规范中滞后的文档。

到目前为止,其他解决方案似乎是:

  • 手动注册和跟踪会话bean
  • 使用JMS /消息驱动Bean(MDB)。

1 个答案:

答案 0 :(得分:2)

在观察者端,只有当前活动的会话才能按照规范进行捕获。如果要在所有会话上进行广播,则需要在注册表中对其进行跟踪。

5.5.6的CDI 2.0:

  

获取Bean的上下文实例,该实例根据Bean的上下文实例声明观察者方法。

上下文实例在6.5.3中定义为

  

当前作用域上下文所服务的实例。