gpt4 book ai didi

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

转载 作者:行者123 更新时间:2023-11-29 03:18:15 25 4
gpt4 key购买 nike

背景

在“普通老式”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/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com