java - 是否可以在不可修改的 ObservableSet 上注册 SetChangeListener?

标签 java set listener observablecollection javafx-8

背景

在“普通老式”JavaFX 桌面应用程序中,我有一个实体,它公开了一个不可修改的 ObservableSet View 。像这样:

public ObservableSet<AbstractMessage> getMessagesUnmodifiable() {
    return FXCollections.unmodifiableObservableSet(messages);
}

在其他地方,客户端代码像这样使用这个 API:

model.getMessagesUnmodifiable()
        .addListener((SetChangeListener<AbstractMessage>) change -> {
            if (change.wasAdded())
                // ... do some work
        });

从我编写客户端代码的第一天(实际上是今天)开始,我立即感到有点担心。是否可以将监听器添加到不可修改 Set?如果我添加一个监听器,那么我是否修改了 Set?

理论

我没有找到关于这个主题的文档。 FXCollections.unmodifiableObservableSet() 的 JavaDoc有一句话说与这个问题无关:

Creates and returns unmodifiable wrapper on top of provided observable set.

表弟FXCollections.unmodifiableObservableMap()还有话要说:

Only mutation operations made to the underlying ObservableMap will be reported to observers that have registered on the unmodifiable instance. This allows clients to track changes in a Map but disallows the ability to modify it.

我在引用中用粗体标记的正是我想要实现的,但使用的是 Set 而不是 Map。由于不可修改和可观察的映射是可能的,我认为它也必须对集契约(Contract)样起作用。

练习

我的客户端代码没有抛出异常。就是这样。但它不起作用。我的听众永远不会被召唤。它确实有效如果我只是为了测试重写实体的 getter 方法,以便他返回可修改的 ObservableSet 引用,上面没有糖。因此,对 FXCollections.unmodifiableObservableSet() 的调用显然以某种方式破坏了集合的行为。

我在一个单独的测试应用程序中快速编写了几行代码。然而,在我的测试应用程序(一个简单的 public static void main)中,它就像一个魅力。我重写了我的测试应用程序和我的真实桌面应用程序,以便两者之间的代码绝对相同,结果如下:向不可修改的 View 添加监听器在我的桌面应用程序中不起作用,但是它在我的测试应用程序中没有缺陷。我无法解释这个发现。

最佳答案

FXCollections.unmodifiableObservableSet() 返回的集合以神秘的方式工作。您可能认为您向支持集添加了一个监听器,但实际上,您添加的监听器直接进入包装器,包装器已将自己注册为支持集的监听器(WeakSetChangeListener) .

只要包装器还活着,他的个人监听器就会从支持集中获取事件,并将这些事件分发给我在包装器中注册的监听器。但是由于我从未存储对包装器的强引用,所以包装器被垃圾收集器拾取并处理掉只是时间问题。我所有喜欢触发的听众都加入了旅程。

这也是我的小型测试应用程序和桌面应用程序之间的全部区别,尽管代码相同,但结果却完全不同。我的测试应用程序立即从头到尾执行完毕。尽管我的桌面应用程序从注册监听器到实际事件需要时间。就在那段时间里,GC 介入并把我搞砸了。

在只花时间阅读源代码之后,从理论上讲,我必须得出结论,不可修改但可观察的 Map 的工作方式相同。所以吸取教训:保存对不可修改的包装器的引用或修改客户端的 API,以便客户端可以将监听器直接传递给支持 Set(这是我选择的解决方案)。

有关弱监听器如何在 JavaFX 中工作的更多信息,请参阅我发布的另一个答案 here .

关于java - 是否可以在不可修改的 ObservableSet 上注册 SetChangeListener?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25224170/

相关文章:

mysql - 设置然后选择

android - 无法在 OnAudioFocusChangeListener 中处理 "duck"我的音频的通知声音

java - 带有 Tiles : java. lang.NoClassDefFoundError 的 Struts 2 应用程序错误:org/apache/tiles/listener/TilesListener

java - 从数组中减去不起作用

java - 测试随机数

java - static 关键字是否会阻止将新对象分配给变量?

java - 在运行用户提供的 Java 代码时,我应该防范哪些安全风险?

java - 从对象获取信息(JAVA)

c# - 在 C# 中设置和获取

java - 多个弹跳球线程问题