JavaFX TableView : Rapid change in items list not reflected

标签 java javafx javafx-8 javafx-tableview

这很难解释,所以我举个例子:

@Override
public void start(Stage primaryStage) throws Exception
{
    final VBox vbox = new VBox();
    final Scene sc = new Scene(vbox);
    primaryStage.setScene(sc);

    final TableView<Person> table = new TableView<>();
    final TableColumn<Person, String> columnName = new TableColumn<Person, String>("Name");
    table.getColumns().add(columnName);

    final ObservableList<Person> list = FXCollections.observableArrayList();
    list.add(new Person("Hello"));
    list.add(new Person("World"));
    Bindings.bindContent(table.getItems(), list);

    columnName.setCellValueFactory(new PropertyValueFactory<>("name"));

    vbox.getChildren().add(table);

    final Button button = new Button("test");
    button.setOnAction(event ->
    {
        final Person removed = list.remove(0);
        removed.setName("Bye");
        list.add(0, removed);
    });
    vbox.getChildren().add(button);


    primaryStage.show();
}

public static class Person
{
    private String name = "";

    public Person(String n)
    {
        name = n;
    }

    public String getName()
    {
        return name;
    }

    public void setName(String n)
    {
        name = n;
    }
}

在这个例子中,我展示了一个 TableView,其中只有一个名为“Name”的列。运行此示例代码,您将得到两行:第一行“Name”列中包含“Hello”;第二行的“名称”列中包含“世界”。

此外,还有一个按钮,此按钮从列表中删除第一个 Person 对象,然后对该对象进行一些更改,然后将其添加回相同的索引处。这样做会导致添加到 ObservableList 的任何 ListChangeListener 被触发,我已经测试过这是真的。

我希望带有“Hello”的行被替换为“Bye”,但 TableView 似乎继续显示“Hello”。如果我在将删除的 Person 对象添加回列表之前使用 TimeLine 添加延迟,它将变为“Bye”。

final Timeline tl = new Timeline(new KeyFrame(Duration.millis(30), ae -> list.add(0, removed)));
tl.play();

API 有什么奇怪的地方吗?有没有办法解决这个问题?

最佳答案

这本质上是预期的行为。

请注意(我猜您正在尝试解决此问题),如果您只是调用

list.get(0).setName("Bye");

在底层数据方面具有相同的效果,表不会更新,因为无法通知元素中的 String 字段 name列表的内容已更改。

代码

Person removed = list.remove(0);
removed.setName("Bye");
list.add(0, removed);

实际上等同于 list.get(0).setName("Bye");:您只是在更改之前暂时从列表中删除该项目,然后再将其添加回去。就列表而言,最终结果是相同的。我猜你这样做是希望从列表中删除和替换项目会说服表注意到项目的状态已经改变。不能保证会是这种情况。这是正在发生的事情:

两个列表之间的绑定(bind):

Bindings.bindContent(table.getItems(), list);

像任何其他绑定(bind)一样工作:它定义如何获取绑定(bind)的值(list 的元素),并在 list 无效时将数据标记为无效随时。后者发生在您从 list 中添加和删除元素时。

TableView 不会在每次对列表的绑定(bind)发生变化时执行布局;相反,当 then 绑定(bind)无效(添加或删除元素)时, TableView 将自身标记为可能需要重绘。然后,在下一个渲染脉冲中,表格将检查数据并查看是否真的需要重绘,并在需要时重新渲染。此实现具有明显的性能节省功能。

那么您的代码发生的情况是从列表中删除了一个项目,导致绑定(bind)被标记为无效。然后更改项目(通过调用 setName(...)),然后将相同的项目添加回列表的相同位置。这也会导致绑定(bind)被标记为无效,这没有任何效果(它已经无效)。

在删除和重新添加此元素之间不会出现渲染脉冲。因此,表第一次实际查看对列表所做的更改必须在整个删除-更改-添加过程之后。届时,该表将看到该列表仍然包含与它之前包含的完全相同的顺序的完全相同的元素。 (其中一个元素的内部状态已经改变,但由于这不是一个可观察的值——不是 JavaFX 属性——表不知道这一点。)因此,表看不到任何变化(或看到所有更改都相互抵消了),并且不会重新呈现。

在您添加暂停的情况下,在移除项目和重新添加项目之间会出现一个(或两个)渲染帧。因此,表格实际上在没有项目的情况下呈现一两个帧,当它被重新添加时,它会重新添加并呈现当前值。 (您可能可以通过暂停 16 或 17 毫秒来使行为不可预测,这恰好是一个渲染帧的时间点。)

不清楚您真正打算做什么。如果您试图在不使用 JavaFX 属性的情况下说服表进行更新,您可以这样做

list.get(0).setName("Bye");
table.refresh();

虽然这不是一个非常令人满意的解决方案。

还要注意

list.remove(0);
list.add(0, new Person("Bye"));

也将起作用(因为现在添加的元素与删除的元素不同)。

更好的方法是使用 JavaFX 属性实现您的模型类:

public static class Person
{
    private final StringProperty name = new SimpleStringProperty("");

    public Person(String n)
    {
        setName(n);
    }

    public StringProperty nameProperty() {
        return name ;
    }

    public final String getName()
    {
        return nameProperty().get();
    }

    public final void setName(String n)
    {
        nameProperty().set(n);
    }
}

然后简单地调用

list.get(0).setName("Bye");

将更新表格(因为单元格将观察该属性)。

关于JavaFX TableView : Rapid change in items list not reflected,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43060044/

相关文章:

java - Intellij 编辑器因 stackoverflow 而卡住

JavaFX 与 Java Swing 在行业中的使用

java - 显示三张卡片图像 - JavaFX

Windows 上的 Javafx 应用程序无需 Java(Maven 或...?)

java - 如何在 java 中随机播放 FilteredList?

java - Spring Framework 4.0.3 BeanDefinitionStoreException 从 ServletContext 资源解析 XML 文档的意外异常

java - 我需要一个正则表达式来匹配 puppet 事实

java - 如何从命令行读取值并在引导注释配置类中使用它们来定义spring bean?

java - 如何使 JavaFX TreeView 和 TreeItem 可序列化?

java - 用于文本输入/javafx 的矩形光标