JavaFX ObservableList 删除侦听器中的元素抛出异常

JavaFX ObservableList remove element in listener throws exception

当我将侦听器附加到可观察列表并在该侦听器中尝试删除一些元素时,在某些情况下它会通过,而在某些情况下会崩溃。

场景: 项目已从列表中删除。它触发侦听器,在该侦听器中我尝试删除另一个项目。

  1. 如果在侦听器中我尝试删除不只是最初删除的元素,它工作正常。
  2. 如果在侦听器中我尝试删除仅次于最初删除的元素,它会因 UnsupportedOperationException 而崩溃!!!

有没有人遇到过类似的问题?您有任何提示、建议或解决方法吗?

我知道你可以同时删除两者,但问题是在监听器中我需要检测我需要删除哪些项目,所以我在那里删除了我。

这是 ObservableList 中的错误吗?

我希望它始终有效,或者至少总是崩溃。

这是代码示例:

import javafx.application.Application;
import javafx.collections.FXCollections;
import javafx.collections.ListChangeListener;
import javafx.collections.ObservableList;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.scene.layout.VBox;
import javafx.stage.Stage;

public class ListRemoveFromListener extends Application {

    boolean changing = false;

    @Override
    public void start(Stage primaryStage) throws Exception {
        VBox vbox = new VBox();

        Button buttonSuccess = new Button("remove success");
        buttonSuccess.setOnAction(e -> {
            removeSuccess();
        });

        Button buttonBreak = new Button("Remove breaks");
        buttonBreak.setOnAction(e -> {
            removeBreaks();
        });

        vbox.getChildren().addAll(buttonSuccess, buttonBreak);

        Scene scene = new Scene(vbox);

        primaryStage.setScene(scene);
        primaryStage.show();
    }

    /**
     * If we try in listener to remove element just next to one that was
     * initially removed, exception is thrown.
     */
    private void removeBreaks() {
        ObservableList<String> list = FXCollections.observableArrayList();
        list.add("first");
        list.add("second");
        list.add("third");
        list.add("fourth");
        list.add("fifth");
        list.add("sixth");

        list.addListener(new ListChangeListener<String>() {

            @Override
            public void onChanged(Change<? extends String> c) {
                list.remove("second");
            }
        });

        list.remove("third");
    }

    /**
     * If we try in listener to remove element that is not just next to initially
     * removed element, element is removed and all works O.
     */
    private void removeSuccess() {
        ObservableList<String> list = FXCollections.observableArrayList();
        list.add("first");
        list.add("second");
        list.add("third");
        list.add("fourth");
        list.add("fifth");
        list.add("sixth");

        list.addListener(new ListChangeListener<String>() {

            @Override
            public void onChanged(Change<? extends String> c) {
                list.remove("fifth");
            }
        });

        list.remove("third");
    }

    public static void main(String[] args) {
        launch(args);
    }

}

ListChangeListener.Change 文档状态:

the source list cannot be modified inside the listener

您可以通过调用 Platform.runLater 来安排在将来某个时间执行的额外更改来解决此问题:

list.addListener((ListChangeListener<String>) c -> {
    if (list.contains("second")) {
        Platform.runLater(() -> list.remove("second"));
    }
});

这样做时要小心,不要导致级联的无限循环更改。