Я добавляю некоторые элементы в ObservableList
в JavaFX
и после добавления некоторых элементов я начинаю получать NullPointerException
для всех элементов, которые я добавляю. Согласно JavaDoc
метод add()
JavaDoc
NullPointerException
когда:
NullPointerException - если указанный элемент равен NULL, и этот список не разрешает нулевые элементы
Но, как вы можете видеть, когда я отлаживаю свой элемент, он НЕ равен нулю:
Итак, почему я получаю это 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). Кажется, я обречен, ничего не могу сделать, пока они не исправят ошибку.
Хорошо, кажется, что ошибка RT-37798 решена в JDK 8u20 Build 21. Будет представлен ошибка, которую мне нужно увидеть. Если я смогу получить ответ (https://javafx-jira.kenai.com/browse/RT-37824),
Спасибо за помощь!
Похоже, что это не data
, а один из компонентов data
который является null
.
Я не смог найти соответствующий исходный код, но если вы посмотрите на
javafx.scene.chart.XYChart$Series$1.onChanged(XYChart.java:1525)
Я уверен, что вы обнаружите разыменование компонента данных.