最初的期望是错误的 - 对于提供的示例。UnsupportedOperationException 发生的原因不同,而不是因为正在“写入”“只读”列表。仍然可以有“只读”列表。我希望下面的答案有助于澄清。
答案需要分两部分来考虑。一:ListProperty 异常和二:只读列表。
1) ListProperty 示例失败,因为没有为该属性分配列表。
这个简化的示例也会引发异常。请注意,删除了所有“只读”方面:
ListProperty<String> lp = new SimpleListProperty<>();
lp.add("element");
这可以通过以下方式纠正:
ObservableList ol = FXCollections.observableArrayList();
ListProperty<String> lp = new SimpleListProperty<>();
lp.setValue(ol);
lp.add("element");
如果我们以类似的方式更改原始示例,那么 ListProperty 和 ObservableList 示例都不会抛出异常,这不是 OP 想要或期望的。
2)第二部分询问为什么可以将元素添加到只读列表中。使用 FXCollections.unmodifiableObservableList 创建只读列表将按预期抛出 UnsupportedOperationException:
ObservableList<String> ol = FXCollections.observableArrayList();
ObservableList<String> uol = FXCollections.unmodifiableObservableList(ol);
uol.add("element");
但这并不能回答为什么 ReadOnlyListWrapper/Property 不这样做的问题?
让我们先处理属性。ListProperty 允许更改值,即它允许您为属性分配不同的列表。ReadOnlyListProperty 不允许这样做,即一旦分配了一个列表,它仍然是该列表对象。列表的内容仍然可以更改。下面的示例对 ReadOnlyListProperty 没有意义:
ObservableList<String> ol1 = FXCollections.observableArrayList();
ObservableList<String> ol2 = FXCollections.observableArrayList();
ListProperty<String> lp = new SimpleListProperty<>(ol1);
lp.setValue(ol2);
所以只读是指属性,而不是列表。
最后 - ReadOnlyListWrapper - 正如 API 文档所述,“此类提供了一个方便的类来定义只读属性。它创建了两个同步的属性。一个属性是只读的,可以传递给外部用户。另一个属性是读取- 并且可写,只能在内部使用。”