为什么在使用System.out.println()打印集合时出现concurrentmodificationexception?

时间:2013-10-27 10:07:44

标签: java concurrentmodification system.out

我正在阅读Java Concurrency in Practice,根据其中的一些java代码,System.out.println()将导致ConcurrentModificationException。代码如下:

private final Set<Integer> set = new HashSet<Integer>();

public synchronized void add(Integer i) {set.add(i); }

public synchronized void remove(Integer i) {set.remove(i);}

public void addTenThings() {

    Random r = new Random();
    for (int i = 0; i < 10; i++) {
        add(r.nextInt());
    }

    System.out.println("DEBUG: add ten elements to " + set );
}

由于System.out.println()方法只会调用toString方法:

public String toString() {
    Iterator<E> i = iterator();
if (! i.hasNext())
    return "[]";

StringBuilder sb = new StringBuilder();
sb.append('[');
for (;;) {
    E e = i.next();
    sb.append(e == this ? "(this Collection)" : e);
    if (! i.hasNext())
    return sb.append(']').toString();
    sb.append(", ");
}
}

我仍然无法理解为什么ConcurrentModificationException会被扔掉?

1 个答案:

答案 0 :(得分:1)

假设有2个线程 - AB同时执行方法addTenThings()。他们可以这样做,因为该方法不同步。

然后,如果线程A正在执行toString()的{​​{1}}方法,从而迭代它,则线程set仍在执行循环,并调用B方法可能导致add(),因为两个线程仅在同一ConcurrentModificationException上运行。没有什么能阻止线程执行set语句,而另一个线程正在该方法中执行add(r.nextInt())语句。