调试的错误就是编程给你最好的东西,因为在每个错误上面都标志着前进的一步。
package list;

import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.ListIterator;

public class listTest2 {

/**
* @param args
*/
public static void main(String[] args) {
List list = new ArrayList();

list.add("java1");
list.add("java2");
list.add("java3");
/*
* 当方法检测到对象的并发修改,但不允许这种修改时,抛出此异常。
*
* 例如,某个线程在 Collection 上进行迭代时,通常不允许另一个线性修改该
* Collection。通常在这些情况下,迭代的结果是不确定的。如果检测到这种行为,一些迭代器实现(包括 JRE 提供的所有通用
* collection 实现)可能选择抛出此异常。执行该操作的迭代器称为快速失败
* 迭代器,因为迭代器很快就完全失败,而不会冒着在将来某个时间任意发生不确定行为的风险。
*
* 注意,此异常不会始终指出对象已经由不同
* 线程并发修改。如果单线程发出违反对象协定的方法调用序列,则该对象可能抛出此异常。例如,如果线程使用快速失败迭代器在
* collection 上迭代时直接修改该 collection,则迭代器将抛出此异常。
*
* 注意,迭代器的快速失败行为无法得到保证,因为一般来说,不可能对是否出现不同步并发修改做出任何硬性保证。快速失败操作会尽最大努力抛出
* ConcurrentModificationException
* 。因此,为提高此类操作的正确性而编写一个依赖于此异常的程序是错误的做法
* ,正确做法是:ConcurrentModificationException 应该仅用于检测 bug。
*/
// 解决方案:在迭代时不要使用集合的方法操作元素。使用迭代器的方法ListIterator
/*
* java.util 接口 ListIterator<E>
* 系列表迭代器,允许程序员按任一方向遍历列表、迭代期间修改列表,并获得迭代器在列表中的当前位置。ListIterator
* 没有当前元素;它的光标位置 始终位于调用 previous() 所返回的元素和调用 next() 所返回的元素之间。长度为 n
* 的列表的迭代器有 n+1 个可能的指针位置,
*/
// 查api,找到List的ListIterator()方法
ListIterator<String> it=list.listIterator();

// Iterator<String> it = list.iterator();
while (it.hasNext()) {
Object obj = it.next();// java.util.ConcurrentModificationException异常

if ("java2".equals(obj)) {// 字符串在前,防止空指针异常。
it.add("Android");

}

}
System.out.println(list);
}

}