ObservableList, выбрасывающий NullPointerException

1

Я добавляю некоторые элементы в ObservableList в JavaFX и после добавления некоторых элементов я начинаю получать NullPointerException для всех элементов, которые я добавляю. Согласно JavaDoc метод add() JavaDoc NullPointerException когда:

NullPointerException - если указанный элемент равен NULL, и этот список не разрешает нулевые элементы

Но, как вы можете видеть, когда я отлаживаю свой элемент, он НЕ равен нулю:

Изображение 174551

Итак, почему я получаю это NullPointerException?

Exception in thread "JavaFX Application Thread" java.lang.NullPointerException
    at javafx.scene.chart.XYChart$Series$1.onChanged(XYChart.java:1525)
    at com.sun.javafx.collections.ListListenerHelper$SingleChange.fireValueChangedEvent(ListListenerHelper.java:158)
    at com.sun.javafx.collections.ListListenerHelper.fireValueChangedEvent(ListListenerHelper.java:72)
    at javafx.collections.ObservableListBase.fireChange(ObservableListBase.java:233)
    at javafx.collections.ListChangeBuilder.commit(ListChangeBuilder.java:482)
    at javafx.collections.ListChangeBuilder.endChange(ListChangeBuilder.java:541)
    at javafx.collections.ObservableListBase.endChange(ObservableListBase.java:205)
    at javafx.collections.ModifiableObservableListBase.add(ModifiableObservableListBase.java:155)
    at java.util.AbstractList.add(AbstractList.java:108)
    at sample.Main$1.handle(Main.java:115)
    at javafx.animation.AnimationTimer$AnimationTimerReceiver$1.run(AnimationTimer.java:58)
    at javafx.animation.AnimationTimer$AnimationTimerReceiver$1.run(AnimationTimer.java:56)
    at java.security.AccessController.doPrivileged(Native Method)
    at javafx.animation.AnimationTimer$AnimationTimerReceiver.handle(AnimationTimer.java:56)
    at com.sun.scenario.animation.AbstractMasterTimer.timePulseImpl(AbstractMasterTimer.java:359)
    at com.sun.scenario.animation.AbstractMasterTimer$MainLoop.run(AbstractMasterTimer.java:269)
    at com.sun.javafx.tk.quantum.QuantumToolkit.pulse(QuantumToolkit.java:475)
    at com.sun.javafx.tk.quantum.QuantumToolkit.pulse(QuantumToolkit.java:460)
    at com.sun.javafx.tk.quantum.QuantumToolkit$13.run(QuantumToolkit.java:327)
    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.access$300(WinApplication.java:39)
    at com.sun.glass.ui.win.WinApplication$4$1.run(WinApplication.java:112)
    at java.lang.Thread.run(Thread.java:745)

Где sample.Main$1.handle(Main.java:115), где я использую метод add().

РЕДАКТИРОВАТЬ:

Кажется, это происходит только тогда, когда я добавляю элементы внутри таких вещей, как TimerTask() или AnimationTimer():

new AnimationTimer(){
            @Override
            public void handle(long now) {
                for (XYChart.Data<Double, Double> data : fullDataQueue){
                    chartData.add(data);
                }
            }
        }.start();

В противном случае я не получу этого искушения!

EDIT 2:

Хорошо, я думаю, что знаю, в чем проблема, вот тестовый код:

@Test
    public void testSeriesAddInAnimator(){
        // Data
        XYChart.Series<Double, Double> series = new XYChart.Series<>();
        ConcurrentLinkedQueue<XYChart.Data<Double, Double>> fullDataQueue = new ConcurrentLinkedQueue<>();
        ObservableList<XYChart.Data<Double, Double>> chartData = FXCollections.observableArrayList();

        series.setData(chartData);

        // Start adding data
        final Random random = new Random();
        for(int n = 0; n < 10000; ++n){
            fullDataQueue.add(new XYChart.Data<>((double)n, random.nextDouble()));
        }

        new AnimationTimer(){
            @Override
            public void handle(long now) {
                for (XYChart.Data<Double, Double> data : fullDataQueue){
                    chartData.add(data);
                }
                System.out.println("Stop!");
                //stop(); -> Uncomment this and we get no exception!
            }
        }.start();
        System.out.println("Done testSeriesAddInAnimator()!");
    }

Это происходит, если я повторно добавлю существующие данные в Серию. Это не ошибка RT-37798, но я думаю, что это тоже ошибка. Если мы прокомментируем метод stop() внутри AnimationTimer те же данные будут добавлены повторно, и мы начнем получать все это NullPointerException.

Исключением является строка 1525 XYChart.java:

for(int i=c.getFrom(); i<c.getTo(); i++) {
                        getData().get(i).setSeries(Series.this);
                        // update linkedList Pointers for data in this series
                        if (begin == null) {
                            begin = getData().get(i);
                            begin.next = null;
                        } else {
                            if (i == 0) {
                                getData().get(0).next = begin;
                                begin = getData().get(0);
                            } else {
                                Data<X,Y> ptr = begin;
                                for (int j = 0; j < i -1 ; j++) {
                                    ptr = ptr.next;  // NPE HERE!!!!!
                                }
                                getData().get(i).next = ptr.next;
                                ptr.next = getData().get(i);
                            }
                        }
                    }

Но если я попытаюсь удалить все элементы в ObservableList а затем добавить те, которые мне нужны, я получаю NullPointerException связанное с указанной вами ошибкой (RT-37798). Кажется, я обречен, ничего не могу сделать, пока они не исправят ошибку.

  • 1
    Взгляните на этот вопрос и ответьте: stackoverflow.com/a/24359013/682840
  • 0
    Итак, еще одна ошибка JavaFX, я рассматриваю возможность перехода на Swing снова ...
Показать ещё 5 комментариев
Теги:
javafx-8
nullpointerexception

2 ответа

0
Лучший ответ

Хорошо, кажется, что ошибка RT-37798 решена в JDK 8u20 Build 21. Будет представлен ошибка, которую мне нужно увидеть. Если я смогу получить ответ (https://javafx-jira.kenai.com/browse/RT-37824),

Спасибо за помощь!

  • 1
    осторожно: исправление в u20 может быть просто косметикой - больше не выбрасывает NPE, но внутренняя структура данных серии может быть не синхронизирована со списком элементов данных
  • 0
    Да, я прочитал ваш комментарий, поэтому решил развернуть собственную библиотеку диаграмм. Спасибо за вашу помощь!
1

Похоже, что это не data, а один из компонентов data который является null.

Я не смог найти соответствующий исходный код, но если вы посмотрите на

javafx.scene.chart.XYChart$Series$1.onChanged(XYChart.java:1525)

Я уверен, что вы обнаружите разыменование компонента данных.

  • 0
    Кажется, это не так, см. Мой отредактированный вопрос!
  • 0
    если это действительно связано с javafx-jira.kenai.com/browse/RT-37798 - как я подозреваю - делать нечего, кроме как ждать исправления ошибки :-( Более глубокая причина - это сильно связанная конструкция / реализация с обратные и обратные обратные вызовы повсюду, и нет четкого (читай: не задокументировано) договора о том, кто за что отвечает и когда.

Ещё вопросы

Сообщество Overcoder
Наверх
Меню