UI 中的 Vaadin 通知与 Akka 演员

Vaadin notification in UI with Akka actor

我正在致力于将基于 akka actor 的后端集成到 vaadin 应用程序中。 我面临的问题是我敢说简单(请原谅我,我是 vaadin 初学者 :P)。

项目结构(简化)
2 个模块前端和后端。

问题
在我的一个观点中,我有一个点击事件处理程序,它通过 UIActor 向其中一个后端 actor 发送消息,一旦计算完成,后者又会回复。
这是 UIActor

中的 "onReceive" 方法
public void onReceive(Object message) throws Exception {

    if (message  == "test") {
        System.out.println("showing notification by "+currentUI);
        currentUI.testMethod();

    } else if (message instanceof MyUI) {
        System.out.println("this is the ui actor test request forwarded");
        currentUI = (MyUI) message;
        backEndAgent.tell("check",getSelf());
    } else {
        unhandled(message);
    }
}

currentUI 包含 (MyUI) getCurrent() 的 return 值,由处理程序作为消息发送,如果您想知道的话。

至于 testMethod,它非常基础。

public void testMethod() {
        Notification.show("Success", Notification.Type.HUMANIZED_MESSAGE);
    }

不幸的是,由于 Page.getCurrent() returning null.

,此调用在 Show 方法中生成了一个空指针

所以我的问题是为什么我在这里得到一个空指针?
同样在我的演员中,我将 currentUI 引用作为消息发送,因为否则调用 UIActor returns 中的 getCurrent 为 null。知道为什么吗?
编辑
这是新的测试方法页面参数是currentUI.getPage()

    public void testMethod(final Page page) {
            this.access(new Runnable() {
                @Override
                public void run() {
                    new Notification("Success",Notification.Type.HUMANIZED_MESSAGE).show(page);

                }
            });
        }

这以某种方式解决了问题,但仅在我第二次单击按钮而不是第一次单击按钮后才显示通知。
这让我发疯。

Notification.show() 使用 Page.getCurrent() 获取当前页面。 Page.getCurrent() 是一个 ThreadLocal,它是在传统的 HTTP 请求-响应周期中设置的,这意味着 Notification.show() 在 HTTP 请求期间有效(return null 无效) -响应线程。

在你的例子中,Notification.show() 是在 HTTP 请求-响应线程之外调用的,这就是它 returns null 的原因。如果我从你的代码中理解正确,currentUI 是对你当前的 UI 的引用,而不是 null。在这种情况下,您可以将代码从

public void testMethod() {
    Notification.show("Success", Notification.Type.HUMANIZED_MESSAGE);
}

public void testMethod() {
    new Notification("Success", Notification.Type.HUMANIZED_MESSAGE).show(getPage());
}

修改后的版本不再使用 Page.getCurrent(),而是从 UI 实例的 getPage() 方法获取当前页面。

编辑。忘记添加与访问 Vaadin 代码的参与者/外部线程相关的重要信息:

从 Vaadin 的角度来看,调用方法来显示通知的 actor 是一个外部线程。 Vaadin 假定对 Vaadin 组件和 Vaadin 相关其他对象的访问是 locked/syncronized 正确的。当使用传统的 HTTP 请求-响应线程时,适当的 locking/syncronization 就位,但当外部线程(例如 actor)访问这些线程时,开发人员必须确保锁定。确保锁定的一种方法是使用 UI.access:

getUI().access(new Runnable() {
    @Override
    public void run() {
        // insert your code accessing Vaadin related objects here
    }
});

您在单击按钮后看到通知的原因是浏览器不知道当演员调用 Notification.show 时有内容要显示。由单击按钮引起的请求将更改从服务器拉到浏览器,然后显示通知。要在没有用户交互(例如单击按钮)的情况下查看参与者/外部线程的更改,应用程序必须使用轮询或推送。要启用轮询,请为您的 ui.

设置 PollInterval