对Collection进行迭代的标准方法是使用Iterater,无论显示的使用还是for-eache与否,当其他线程可能并发修改容器时,需要在迭代器具对容器加锁。
对于for-each语法,javac生成的代码内在的使用一个Iterator,调用hasNext和next对List进行迭代。在进行迭代时,当容器察觉到在迭代开始后被修改,会抛出一个未检查的ConcurrentModificationException。为了避免出现该情况,需要在迭代期间持有一个容器锁。
List<Widege> widgetList = Collections.synchronizedList(new ArrayList<Widget>);
synchronized(widgetList) {
for(Widegt w : widgetList)
doSomething(w);
}
另外一个办法是复制容器。没有其他的线程能够在迭代期间对其进行修改,但是这样做会有明显的性能开销,取决于多种因素,包括容器大小、响应性。