我正在阅读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
会被扔掉?
答案 0 :(得分:1)
假设有2个线程 - A
和B
同时执行方法addTenThings()
。他们可以这样做,因为该方法不同步。
然后,如果线程A
正在执行toString()
的{{1}}方法,从而迭代它,则线程set
仍在执行循环,并调用B
方法可能导致add()
,因为两个线程仅在同一ConcurrentModificationException
上运行。没有什么能阻止线程执行set
语句,而另一个线程正在该方法中执行add(r.nextInt())
语句。