JavaFX Bindings.size()在发生某些事件后停止工作

2022-03-17 00:00:00 java-8 java javafx javafx-8 javafx-2

我编写了一个小型的、可运行的JavaFX应用程序,它有两种方式来监听ObservableList的大小变化。第一个监听器(第52-58行)工作正常。第二个监听器(第60-66行)在发生某些事件后停止工作。

您可以重现此错误(?)当您经常点击"添加按钮"时。第一次视图单击时,两条消息都被打印出来,之后的一些单击只有第一个监听程序进一步工作。

可运行示例:

import javafx.application.Application;
import javafx.application.Platform;
import javafx.beans.binding.Bindings;
import javafx.beans.value.ChangeListener;
import javafx.beans.value.ObservableValue;
import javafx.collections.FXCollections;
import javafx.collections.ListChangeListener;
import javafx.collections.ObservableList;
import javafx.event.ActionEvent;
import javafx.event.EventHandler;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.scene.control.ListView;
import javafx.scene.layout.StackPane;
import javafx.scene.layout.VBox;
import javafx.stage.Stage;


public class Main extends Application {
    public static void main(String[] args) {
        launch(args);
    }

    @Override
    public void start(Stage primaryStage) {
        final ObservableList<String> list = FXCollections.observableArrayList();
        primaryStage.setTitle("Demonstrator");

        // Button
        Button addButton = new Button();
        addButton.setText("Add Element");
        addButton.setOnAction(new EventHandler<ActionEvent>() {
            @Override
            public void handle(ActionEvent event) {
                list.add("TEST");
            }
        });

        // ListView
        ListView<String> lv = new ListView<String>();
        lv.setItems(list);

        // Add elements to root
        VBox root = new VBox();
        root.getChildren().add(addButton);
        root.getChildren().add(lv);

        // Show scene
        primaryStage.setScene(new Scene(root, 300, 250));
        primaryStage.show();

        // This listener works correct
        list.addListener(new ListChangeListener<String>() {
            @Override
            public void onChanged(Change<? extends String> c) {
                System.out.println("#listener1: " + list.size());
            }
        });

        // This listener stops working after 10-30 clicks on the button
        Bindings.size(list).addListener(new ChangeListener<Number>() {
            @Override
            public void changed(ObservableValue<? extends Number> observable, Number oldValue, Number newValue) {
                System.out.println("#listener2: " + newValue);
            }
        });
    }
}

我在jdk-8u112-windows-x64和jdk-8u112-windows-i586上试用过

有没有人知道我的代码是否有错误,或者是否真的是JavaFX有问题?

我得到了解决方案:

绑定已从垃圾回收器中删除。您必须将绑定存储为字段。

谢谢!


解决方案

行61执行之后。在Bindings.size()中创建的实例不能被任何东西强访问,因此它有资格进行垃圾收集。它可以通过侦听器软访问,但这不会阻止垃圾回收。

所以最终它将被垃圾收集并停止工作。

解决方案是:使其强可达,例如,通过将其存储在类的字段中。

相关文章