代码块
Java
 
 
 
 
  1. Exception in thread "main" java.util.ConcurrentModificationException
  1. at java.util.ArrayList$Itr.checkForComodification(ArrayList.java:909)
  1. at java.util.ArrayList$Itr.next(ArrayList.java:859)
  1. at com.mashibing.tank.Test2.main(Test2.java:14)
 

ArrayList在迭代删除元素时,如果使用不当会发生concurrentModification 异常,常见的错误使用场景

代码块
Java
 
 
 
 
  1. Iterator<Integer> iterator = list.iterator();
  1. while(iterator.hasNext()){
  1. Integer next = iterator.next();
  1. list.remove(next);
  1. }
 
代码块
Java
 
 
 
 
  1. for (Integer next:list
  1. ) {
  1. list.remove(next);
  1. }
 

发生异常原因:

首先明确以上两种方式是等效的,原因:增强for在反编译后语句

代码块
Java
 
 
 
 
  1. Iterator var4 = var1.iterator();
  1. while(var4.hasNext()) {
  1. Integer var3 = (Integer)var4.next();
  1. var1.remove(var3);
  1. }
 

然后我们去分析发生concurrentModification原因:

由于异常发生在 :java.util.ArrayList$Itr.checkForComodification(ArrayList.java:909),我们先去看下此处代码

发现remove(intxOrObject) 方法为 ArrayList 内部类Itr的一个方法 ,当modCount != expectedModCount 时会抛出异常ConcurrentModificationException。

先来解释下概念:

modCount:为 成员变量 list 被修改的次数 ,每次add ,remove都会+1 (The number of times this list has been )

代码块
Java
 
 
 
 
  1. public abstract class AbstractList<E> extends AbstractCollection<E> implements List<E>{
  1. //The number of times this list has been
  1. protected transient int modCount = 0;
 

expectedModCount:为listIterator 方法 局部变量,表示对ArrayList修改次数的期望值,它的初始值为modCount

代码块
Java
 
 
 
 
 
  1. public Iterator<E> iterator() {
  1. return listIterator();
  1. }
  1. public ListIterator<E> listIterator(final int index) {
  1. checkForComodification();
  1. rangeCheckForAdd(index);
  1. final int offset = this.offset;
  1. return new ListIterator<E>() {
  1. int cursor = index;
  1. int lastRet = -1;
  1. //注意这里
  1. int expectedModCount = ArrayList.this.modCount;
  1. .......
  1. }
  1. ....
  1. }
  1. .....
  1. }
 
代码块
Java
 
 
 
 
 
  1. private class Itr implements Iterator<E> {
  1. int cursor; // index of next element to return
  1. int lastRet = -1; // index of last element returned; -1 if no such
  1. int expectedModCount = modCount;
  1. Itr() {}
  1. public boolean hasNext() {
  1. return cursor != size;
  1. }
  1. @SuppressWarnings("unchecked")
  1. public E next() {
  1. checkForComodification();
  1. int i = cursor;
  1. if (i >= size)
  1. throw new NoSuchElementException();
  1. Object[] elementData = ArrayList.this.elementData;
  1. if (i >= elementData.length)
  1. throw new ConcurrentModificationException();
  1. cursor = i + 1;
  1. return (E) elementData[lastRet = i];
  1. }
  1. }
 

这里假设原始list 长度为10(只做过新增,没有做过删除操作) 。

这时候最初 AbstractList.modCount =10 ;

Iterator.expectedModCount = modCount =10 ;

1.当第一次循环调用next时,无变动list ,所以expectedModCount = modCount =10 ,

2.当 arrayList.remove(indexOrObject)时,modCount++ 变为11 ,expectedModCount 依然为10 。

3.第二次调用next 时,Itr 会执行checkForComodification() ,也就是判断expectedModCount、 modCount 是否一致,如果不一致则抛出ConcurrentModificationException 。

正确姿势:

关注点:

1.增强for循环不要删除元素

2.使用Iterator<Integer> iterator = list.iterator(); 方式时,先获取元素,然后使用iterator.remove() 进行删除

代码块
Java
 
 
 
 
  1. Iterator<Integer> iterator = list.iterator();
  1. while (iterator.hasNext() ) {
  1. System.out.println(iterator.next());
  1. iterator.remove();
  1. }
  1. for (int i = 0; i < list.size(); i++) {
  1. System.out.println(list.get(i));
  1. list.remove(i);
  1. }
 

为什么Iterator<Integer> iterator = list.iterator(); 方式时 用iterator remove 不会出现问题?看源码:

代码块
Java
 
 
 
 
  1. private class Itr implements Iterator<E> {
  1. public void remove() {
  1. if (lastRet < 0)
  1. throw new IllegalStateException();
  1. checkForComodification();
  1. try {
  1. ArrayList.this.remove(lastRet);
  1. cursor = lastRet;
  1. lastRet = -1;
  1. //关注这里, Itr里将modCount 赋值给expectedModCount
  1. expectedModCount = modCount;
  1. } catch (IndexOutOfBoundsException ex) {
  1. throw new ConcurrentModificationException();
  1. }
  1. }
 

为什么普通循环不会有ConcurrentModificationException ,看源码:

代码块
Java
 
 
 
 
  1. public class ArrayList<E> extends AbstractList<E>
  1. implements List<E>, RandomAccess, Cloneable, java.io.Serializable
  1. {
  1. public boolean remove(Object o) {
  1. if (o == null) {
  1. for (int index = 0; index < size; index++)
  1. if (elementData[index] == null) {
  1. fastRemove(index);
  1. return true;
  1. }
  1. } else {
  1. for (int index = 0; index < size; index++)
  1. if (o.equals(elementData[index])) {
  1. fastRemove(index);
  1. return true;
  1. }
  1. }
  1. return false;
  1. }
  1. private void fastRemove(int index) {
  1. modCount++;
  1. int numMoved = size - index - 1;
  1. if (numMoved > 0)
  1. System.arraycopy(elementData, index+1, elementData, index,
  1. numMoved);
  1. elementData[--size] = null; // clear to let GC do its work
  1. }
  1. }
 

List remove ConcurrentModificationException源码分析的更多相关文章

  1. 集合操作出现的ConcurrentModificationException(源码分析)

    摘要: 为了保证线程安全,在迭代器迭代的过程中,线程是不能对集合本身进行操作(修改,删除,增加)的,否则会抛出ConcurrentModificationException的异常. 示例: publi ...

  2. Java并发-ConcurrentModificationException原因源码分析与解决办法

    一.异常原因与异常源码分析 对集合(List.Set.Map)迭代时对其进行修改就会出现java.util.ConcurrentModificationException异常.这里以ArrayList ...

  3. HashMap与TreeMap源码分析

    1. 引言     在红黑树--算法导论(15)中学习了红黑树的原理.本来打算自己来试着实现一下,然而在看了JDK(1.8.0)TreeMap的源码后恍然发现原来它就是利用红黑树实现的(很惭愧学了Ja ...

  4. 【集合框架】JDK1.8源码分析之HashMap & LinkedHashMap迭代器(三)

    一.前言 在遍历HashMap与LinkedHashMap时,我们通常都会使用到迭代器,而HashMap的迭代器与LinkedHashMap迭代器是如何工作的呢?下面我们来一起分析分析. 二.迭代器继 ...

  5. 【JUC】JDK1.8源码分析之CopyOnWriteArrayList(六)

    一.前言 由于Deque与Queue有很大的相似性,Deque为双端队列,队列头部和尾部都可以进行入队列和出队列的操作,所以不再介绍Deque,感兴趣的读者可以自行阅读源码,相信偶了Queue源码的分 ...

  6. 【JUC】JDK1.8源码分析之ConcurrentSkipListSet(八)

    一.前言 分析完了CopyOnWriteArraySet后,继续分析Set集合在JUC框架下的另一个集合,ConcurrentSkipListSet,ConcurrentSkipListSet一个基于 ...

  7. HashMap实现原理及源码分析

    哈希表(hash table)也叫散列表,是一种非常重要的数据结构,应用场景及其丰富,许多缓存技术(比如memcached)的核心其实就是在内存中维护一张大的哈希表,而HashMap的实现原理也常常出 ...

  8. JDK Collection 源码分析(2)—— List

    JDK List源码分析 List接口定义了有序集合(序列).在Collection的基础上,增加了可以通过下标索引访问,以及线性查找等功能. 整体类结构 1.AbstractList   该类作为L ...

  9. [源码解析]HashMap和HashTable的区别(源码分析解读)

    前言: 又是一个大好的周末, 可惜今天起来有点晚, 扒开HashMap和HashTable, 看看他们到底有什么区别吧. 先来一段比较拗口的定义: Hashtable 的实例有两个参数影响其性能:初始 ...

随机推荐

  1. Python 装饰器 多装饰器同时装饰一个函数 多参数函数

    装饰器是在不修改源代码的情况下,使用装饰器增加原函数的功能. 在软件开发中有一个原则——"开放-封闭",简单地说就是已经实现的功能不允许被修改,但可以被扩展. 封闭:已经实现的功能 ...

  2. python自动化测试之函数(匿名函数lambda和三目运算等(高级用法))

    ''' 匿名函数: lambda ''' def Add(a,b): print(a+b) Add(2,3) per = lambda a,b:a+b print(per(2,3)) ''' 三目运算 ...

  3. python djangjo 文件上传

    视图: request.GET 获取数据 request.POST   提交数据 request.FILES  获取文件用 checkbox 等多选的内容   request.POST.getlist ...

  4. SpringMVC_Day01

    项目结构 //SpringMVC配置文件 <?xml version="1.0" encoding="UTF-8"?> <!-- spring ...

  5. 通过zxing生成二维码

    二维码现在随处可见,在日常的开发中,也会经常涉及到二维码的生成,特别是开发一些活动或者推广方面的功能时,二维码甚至成为必备功能点.本文介绍通过 google 的 zxing 包生成带 logo 的二维 ...

  6. Spotlight 监控工具使用

    监控MySQL数据库性能的工具:Spotlight on MySQL    <转载> 我们的服务器数据库:是在windows2003上. 这款工具非常的花哨,界面很漂亮,自带报警. 1.创 ...

  7. python3下BeautifulSoup练习一(爬取小说)

    上次写博客还是两个月以前的事,今天闲来无事,决定把以前刚接触python爬虫时的一个想法付诸行动:就是从网站上爬取小说,这样可以省下好多流量(^_^). 因为只是闲暇之余写的,还望各位看官海涵:不足之 ...

  8. doctrine queryBuilder

    为了能够方便的切换数据库,我们有必要使用doctrine的queryBuilder, 但是估计很多人都是不喜欢的(我也是),之前尝试用的时候,发现在doctrine定义的SELECT语法中并没有CON ...

  9. mysql 分表实现方法详解

    如果你需要进行mysql分表了我们就证明你数据库比较大了,就是把一张表分成N多个小表,分表后,单表的并发能力提高了,磁盘I/O性能也提高了.并发能力为什么提高了呢,因为查寻一次所花的时间变短了,如果出 ...

  10. AndroidImageSlider

    最核心的类是SliderLayout,他继承自相对布局,包含了可以左右滑动的SliderView,以及页面指示器PagerIndicator.这两部分都可以自定义. AndroidImageSlide ...