私のコードはTreeItem<String>
バックグラウンド タスクで作成します。これは、多数のコードがあり、それらの作成にかなりの時間がかかり、アプリケーションがフリーズするためです。この例ではあまり意味がありませんが、実際のアプリケーションで遭遇した問題を示しています。ノードを展開すると、プログラムは ConcurrentModificationException をスローします。
jdk1.7.0_17 と JavaFX 2.2.7 を使用しています
Tree
スレッドセーフを作成する方法や問題を回避する方法を知っている人はいますか?
例外
java.util.ConcurrentModificationException
at java.util.ArrayList$Itr.checkForComodification(ArrayList.java:819)
at java.util.ArrayList$Itr.next(ArrayList.java:791)
at com.sun.javafx.collections.ObservableListWrapper$ObservableListIterator.next(ObservableListWrapper.java:681)
at javafx.scene.control.TreeItem.updateExpandedDescendentCount(TreeItem.java:788)
...
コード
import javafx.application.Application;
import javafx.collections.ObservableList;
import javafx.concurrent.Task;
import javafx.scene.Scene;
import javafx.scene.control.TreeItem;
import javafx.scene.control.TreeView;
import javafx.scene.layout.HBox;
import javafx.stage.Stage;
import java.security.SecureRandom;
import java.util.Random;
public class ConcurrentExample extends Application {
public static void main(String[] args) {
launch(args);
}
@Override
public void start(Stage stage) throws Exception {
TreeView<String> treeView = new TreeView<String>(createNode("root"));
HBox hBox = new HBox();
hBox.getChildren().addAll(treeView);
Scene scene = new Scene(hBox);
stage.setScene(scene);
stage.show();
}
Random r = new SecureRandom();
public TreeItem<String> createNode(final String b) {
return new TreeItem<String>(b) {
private boolean isLeaf;
private boolean isFirstTimeChildren = true;
private boolean isFirstTimeLeaf = true;
@Override
public ObservableList<TreeItem<String>> getChildren() {
if (isFirstTimeChildren) {
isFirstTimeChildren = false;
buildChildren(super.getChildren());
}
return super.getChildren();
}
@Override
public boolean isLeaf() {
if (isFirstTimeLeaf) {
isFirstTimeLeaf = false;
isLeaf = r.nextBoolean() && r.nextBoolean() && r.nextBoolean();
}
return isLeaf;
}
private void buildChildren(final ObservableList<TreeItem<String>> children) {
if (!this.isLeaf()) {
Task<Integer> task = new Task<Integer>() {
@Override
protected Integer call() throws Exception {
int i;
int max = r.nextInt(500);
for (i = 0; i <= max; i++) {
children.addAll(new TreeItem[]{createNode("#" + r.nextInt())});
}
return i;
}
};
new Thread(task).start();
}
}
};
}
}