Java缓存使用ArrayList/HashMap的常见问题----ConcurrentModification

对于一些经常读取,但是很少写的数据,经常会使用缓存来存储。常见的做法是:

List list = new ArrayList();
Map map = new HashMap();
//if modify
public void modify() {
     synchronized(list){
         //do modify list
     } 
     synchronized(map){
        //do modify map
     }
}

这样的做法会有一些风险。因为list/map可能在另一个地方正在被迭代使用,比如:

Iterator it = list.iterator();
while(it.hasNext()){ 
       //do things
}
for(String str : it){
      //do things
}

这个时候,会立即报ConcurrentModification的错误,而且错误发生的概率比较小,但是一旦出错,就会造成严重错误以致宕机,因为你肯定不会在迭代的地方try/catch该异常。原因是ArrayList和hashmap这些jdk1.5以下的集合使用的都是强一致性迭代器,见jdk描述:

he iterators returned by all of this class's "collection view methods" are fail-fast: if the map is structurally modified at any time after the iterator is created, in any way except through the iterator's own remove method, the iterator will throw a ConcurrentModification

然后它的建议是使用:

List list = Collections.synchronizedList(new ArrayList(...));
任何时候都有可能使用当前集合的迭代器,所以必须在初始化的时候就同步好所有方法,但这样不是就等于退化到Hashtable了吗?其实就是没有高效的解决办法。

为了解决这个问题,在jdk1.5以后的concurrent集合里面,所有的实现都使用了弱一致性迭代器,不会抛ConcurrentModification错误。

所以,尽量使用ConcurrentHashMap和CopyOnWriteArrayList。他们使用了细粒度锁或者直接读写分离,非常适合读多写少的场景。

你可能感兴趣的:(Java缓存使用ArrayList/HashMap的常见问题----ConcurrentModification)