JavaFX 图表系列 ConcurrentModificationException

标签 java javafx task

我正在 JavaFX 上构建遗传算法,我想要一个图表来更新每一代的平均值和最大适应度。 我有一个按钮,它执行 20 代,每次迭代都会更新图表以及进度条。

过去的代码比较简单,没有那些重复和引用。这些只是关于如何避免异常的想法。但没有任何作用

我的意思是,我什至不迭代或删除某些内容。这只是添加。

我得到这个异常:

Exception in thread "JavaFX Application Thread" java.util.ConcurrentModificationException
    at java.util.ArrayList$Itr.checkForComodification(ArrayList.java:901)
    at java.util.ArrayList$Itr.next(ArrayList.java:851)
    at java.util.Collections$UnmodifiableCollection$1.next(Collections.java:1042)
    at javafx.scene.chart.AreaChart.layoutPlotChildren(AreaChart.java:451)
    at javafx.scene.chart.XYChart.layoutChartChildren(XYChart.java:731)
    at javafx.scene.chart.Chart$1.layoutChildren(Chart.java:94)
    at javafx.scene.Parent.layout(Parent.java:1079)
    at javafx.scene.Parent.layout(Parent.java:1085)
    at javafx.scene.Parent.layout(Parent.java:1085)
    at javafx.scene.Parent.layout(Parent.java:1085)
    at javafx.scene.Scene.doLayoutPass(Scene.java:552)
    at javafx.scene.Scene$ScenePulseListener.pulse(Scene.java:2397)
    at com.sun.javafx.tk.Toolkit.lambda$runPulse$31(Toolkit.java:348)
    at java.security.AccessController.doPrivileged(Native Method)
    at com.sun.javafx.tk.Toolkit.runPulse(Toolkit.java:347)
    at com.sun.javafx.tk.Toolkit.firePulse(Toolkit.java:374)
    at com.sun.javafx.tk.quantum.QuantumToolkit.pulse(QuantumToolkit.java:510)
    at com.sun.javafx.tk.quantum.QuantumToolkit.pulse(QuantumToolkit.java:490)
    at com.sun.javafx.tk.quantum.QuantumToolkit.lambda$runToolkit$405(QuantumToolkit.java:319)
    at com.sun.glass.ui.InvokeLaterDispatcher$Future.run(InvokeLaterDispatcher.java:95)
    at com.sun.glass.ui.win.WinApplication._runLoop(Native Method)
    at com.sun.glass.ui.win.WinApplication.lambda$null$149(WinApplication.java:191)
    at java.lang.Thread.run(Thread.java:745)

这是准备工作

public void prepareChart() {
    final NumberAxis xAxis = new NumberAxis();
    final NumberAxis yAxis = new NumberAxis();
    final AreaChart<Number, Number> graph = new AreaChart<Number, Number>(xAxis, yAxis);
    graph.setCreateSymbols(false);
    graph.setTitle("Fitness - Generations relation");
    graph.getXAxis().setLabel("Generations");
    graph.getYAxis().setLabel("Fitness");

    final Series<Number, Number> averageFitness = new XYChart.Series<Number, Number>();
    averageFitness.setName("Average fitness");

    final Series<Number, Number> maximumFitness = new XYChart.Series<Number, Number>();
    maximumFitness.setName("Maximum fitness");
    graph.getData().addAll(averageFitness, maximumFitness);
    graphBox.getChildren().add(graph);
    graphRef = graph;
}

这是我添加数据的方法

public void addData(double avgFitness, double maxFitness) {
    final Series<Number, Number> averageFitness = graphRef.getData().get(0);
    final Series<Number, Number> maximumFitness = graphRef.getData().get(1);
    averageFitness.getData().add(new XYChart.Data<Number, Number>(generation, avgFitness));
    maximumFitness.getData().add(new XYChart.Data<Number, Number>(generation, maxFitness));
    generation++;

}

以下是单击按钮时发生的情况

public void twentyGenerationsPressed() {

    Task<Void> task = new Task<Void>() {
        @Override
        public Void call() throws InterruptedException {
        for (int i = 1; i <= 20; i++) {
            disableButtons();
            solver.nextGeneration();
            updateProgress(i, 20);
            executor.execute(() -> {
            addData(solver.getAverage(), solver.getCurrentBest().getFitness());
            });

        }

        return null;
        }
    };
    task.setOnSucceeded(e -> {
        updateLabels();
        enableButtons();
    });

    progressBar.progressProperty().bind(task.progressProperty());

    Thread th = new Thread(task);
    th.setDaemon(true);
    th.start();

}

最佳答案

根据经验,任何时候修改 JavaFX 对象的内容,都应该在 JavaFX 应用程序线程上执行。

替换这个:

executor.execute(() -> {
    addData(solver.getAverage(), solver.getCurrentBest().getFitness());
});

这样:

final double avg = solver.getAverage();
final double best = solver.getCurrentBest().getFitness();
Platform.runLater(new Runnable() {
    @Override
    public void run() {
        addData(avg, best);
    }
});

关于JavaFX 图表系列 ConcurrentModificationException,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36988499/

相关文章:

java - 在 JavaFX 中添加/删除动画

process - 线程/进程/任务之间有什么区别?

c# - 对异步任务使用取消

python - 树莓派 : How can I kill an auto-started Tkinter UI?

java - 使用Java SecurityManager阻止我读取文件

java - 在线 java 代码评估器找不到我的变量

java - 对象类型方法的 assertequal

JavaFX TitledPane 双边框使用 css 文件

java - 如何在 JavaFX 中显示我自己的线程中的新舞台?

java - 如何使用 IntelliJ IDEA 导出 JAR(JavaFX 和 Maven)