Java中集合删除元素时候关于ConcurrentModificationException的迷惑点
下面的示例来至于阿里巴巴Java开发手册的集合处理部分的第7条:

运行如下代码,会发现正确运行。
public static void hasNotExcption() {
List<String> list1 = new ArrayList<String>();
list1.add("1");
list1.add("2");
for (String item : list1) {
System.out.println("item : "+item);
if ("1".equals(item)) {
list1.remove(item);
}
}
}
但是运行如下代码,则异常:java.util.ConcurrentModificationException(和1中的代码区别是上面移除判断条件是1,下面的判断条件是2)
public static void hasExcption() {
List<String> list1 = new ArrayList<String>();
list1.add("1");
list1.add("2");
for (String item : list1) {
System.out.println("item : "+item);
if ("2".equals(item)) {
list1.remove(item);
}
}
}
再看如下代码示例,运行结果见注释:
/**
*没有异常
*/
public static void hasNotExcption() {
List<String> list1 = new ArrayList<String>();
list1.add("1");
list1.add("2");
list1.add("3");
list1.add("4"); for (String item : list1) { System.out.println("item : " + item);
// 移除1和3时候会抛ConcurrentModificationException异常
// 但是移除2的时候不会抛出异常
if ("3".equals(item)) {
list1.remove(item);
}
} } /**
* 有异常
*/
public static void hasExcption1() {
List<String> list1 = new ArrayList<String>();
list1.add("1");
list1.add("2");
list1.add("3"); for (String item : list1) {// 增强的for循环底层实现使用的是迭代器
if ("1".equals(item)) {
// 移除并修改“modCount变量”,导致下次遍历时候异常
list1.remove(item);
}
} } /**
* 有异常
*/
public static void hasExcption2() {
List<String> list1 = new ArrayList<String>();
list1.add("1");
list1.add("2");
list1.add("3"); for (String item : list1) {
if ("3".equals(item)) {
list1.remove(item);
}
} }
通过上面三个例子是不是发现了结论?只要移除的是倒数第二个元素的话,就不会发生异常!的确是,移除倒数第二个元素的话就不会异常,那究竟是为什么呢?
知识点1:关于增强for循环的实现
集合的增强for循环实现内部使用的是迭代器,可以通过eclipse的F5调试跟踪。
List<String> list1 = new ArrayList<String>();
list1.add("1");
list1.add("2");
list1.add("3");
for (String item : list1) {
if ("3".equals(item)) {
list1.remove(item);
}
}
知识点2:集合如果判定的是并发修改错误?
细心的话可以通过异常发现异常的判断代码是:
if (modCount != expectedModCount)
throw new ConcurrentModificationException();
其中的modCount是记录集合的修改次数,而expectedModCount是记录的预期的修改次数,当集合的修改次数和预期修改次数不一致的时候则发生异常。
那么什么时候会修改modCount呢?当我们对集合进行删除或者添加元素时候此时记录集合的修改次数就会发生增加,但是期望的修改次数不会变化,当检查判断异常条件时候就会根据条件处理。
到此回归到上面问题,对于hasExcption1和hasExcption2是如何发生异常的?本例对hasExcption1进行详细解释:
/**
* 有异常
*/
public static void hasExcption1() {
List<String> list1 = new ArrayList<String>();
list1.add("1");
list1.add("2");
list1.add("3");
// 增强的for循环底层实现使用的是迭代器。所以每一次都是调用一个hasNext,然后在调用next方法返回元素给item
for (String item : list1) { if ("1".equals(item)) {
// 移除并修改“modCount变量”,导致下次遍历时候异常
list1.remove(item);
}
} }
增强的for循环底层实现使用的是迭代器。所以每一次都是调用一个hasNext,然后在调用next方法返回元素给item。当我们删除元素之后,下一次循环时候会先调用hasNext是否还有元素,此时还有。在调用next方法返回该元素给item,但是调用next方法时候会检测
集合修改次数和预期修改次数是否相等,如果不等的话则抛出异常。(本例是不等的,因为移除1之后modCount=4,而expectModCount=3)。
那么为什么删除倒数第二个元素不会异常呢?示例代码:
/**
*没有异常
*/
public static void hasNotExcption() {
List<String> list1 = new ArrayList<String>();
list1.add("1");
list1.add("2");
list1.add("3");
list1.add("4"); for (String item : list1) {
if ("3".equals(item)) {
list1.remove(item);
}
} }
因为我们删除倒数第二个元素时候,此时size=size-1,当删除完元素进入下一次循环时候,此时hasNext方法判断是否还有元素的时候返回是false(hasNext使用的是游标和size进行比较)。所以就不会调用next方法了。上面例子我们知道异常就是在next方法中抛的,所以
删除倒数第二个元素就不会有异常。
注意:如果删除的是倒数第二个元素那么最后的结果是否正确呢?答案:显然是不正确的。
通过如下示例代码检验:
/**
*没有异常
*/
public static void hasNotExcption() {
List<String> list1 = new ArrayList<String>();
list1.add("1");
list1.add("2");
list1.add("3");
list1.add("3"); for (String item : list1) {
if ("3".equals(item)) {
list1.remove(item);
}
}
// 显然最后一个3没有删除,所以结果错误
System.out.println(list1); //[1, 2, 3] }
结果是错误的,原因:因为最后一个元素由于在移除倒数第二个元素时候将size-1了,所以遍历集合提前结束了。所以没有完全移除掉所有的“3”
Java中集合删除元素时候关于ConcurrentModificationException的迷惑点的更多相关文章
- java操作数组转list集合删除元素报错ConcurrentModificationException
public static void main(String[] args) { ArrayList<String> list = new ArrayList<String>( ...
- JAVA中循环删除list中元素的方法总结【转】
印象中循环删除list中的元素使用for循环的方式是有问题的,但是可以使用增强的for循环,然后今天在使用时发现报错了,然后去科普了一下,再然后发现这是一个误区.下面就来讲一讲..伸手党可直接跳至文末 ...
- java中循环删除list中元素的方法
重点哈 印象中循环删除list中的元素使用for循环的方式是有问题的,但是可以使用增强的for循环,然后今天在使用时发现报错了,然后去科普了一下,再然后发现这是一个误区.下面就来讲一讲..伸手党可直接 ...
- Java中集合的概述
一.集合和数组的区别 1.数组(可以存储基本数据类型)是用来存现对象的一种容器,但是数组的长度固定,不适合在对象数量未知的情况下使用. 2.集合(只能存储对象,对象类型可以不一样)的长度可变,可在多数 ...
- Java中集合List,Map和Set的区别
Java中集合List,Map和Set的区别 1.List和Set的父接口是Collection,而Map不是 2.List中的元素是有序的,可以重复的 3.Map是Key-Value映射关系,且Ke ...
- JAVA中集合转数组遍历
JAVA中集合的遍历的一种方法时集合转数组遍历,也是就调用Collection中的toArray(). 代码: public static void main(String[] args) { ...
- Java中集合List,Map和Set的差别
Java中集合List,Map和Set的差别 1.List和Set的父接口是Collection.而Map不是 2.List中的元素是有序的,能够反复的 3.Map是Key-Value映射关系,且Ke ...
- 遍历List集合删除元素的出现报错
遍历List集合删除元素的出现报错 遍历List集合删除元素的时候会发生索引越界异常或内容遍历不全等问题. 例子: List<String> al = new ArrayList< ...
- js 遍历集合删除元素
js 遍历集合删除元素 /** * 有效的方式 - 改变下标,控制遍历 */ for (var i = 0; i < arr.length; i++) { if (...) { arr.spli ...
随机推荐
- SpringBoot入门之Thymeleaf的使用
在.net的MVC3 或更高版本等支持 Razor 的框架里使用cshtml,Razor是一种简单的编程语法,用于在网页中嵌入服务器端代码.在使用springboot开发mvc时也有与.net类似的视 ...
- .Net敏捷开发框架6.1.6.2版本,联系QQ:6539471
演示地址:www.fishcmonkey.com .NET敏捷开发框架 6.1.6.2 版本发布 新增手机流程-我的流程(可查看流程进度和表单内容) 新增手机流程-待办任务(可查看流程进度和表单内容, ...
- c# 后台拼接分页Html
public static string ReplaceStr(string originalStr, string oldStr, string newStr) { if (string.IsNul ...
- Contest2075 - 湖南多校对抗(csu1576)大数 Catalan Square
Problem C: Catalan Square Time Limit: 1 Sec Memory Limit: 128 MBSubmit: 42 Solved: 16[Submit][Stat ...
- MEF 插件式开发之 DotNetCore 初体验
背景叙述 在传统的基于 .Net Framework 框架下进行的 MEF 开发,大多是使用 MEF 1,对应的命名空间是 System.ComponentModel.Composition.在 Do ...
- 使用Via浏览器+ADM下载器突破百度网盘下载限速
1.下载必要工具 via浏览器 ADM下载器 2.自定义 UA UA 是一串特殊字符,用来告诉所访问的网站,手机使用的操作系统及版本.CPU 类型.浏览器及版本等信息.UA 内容如下: Mozilla ...
- 使用 json-server 模拟数据
1. 先安装 npm install json-server -g 2.查看是否安装成功 json-server -h 3.准备数据,新建一个文件夹 mock,cd mock,在mock下 新建tes ...
- js之搜索框
目标效果:点击搜索框,搜索框内提示信息消失,可输入搜索信息,点击搜索框外搜索框如果没提示信息或者为空时,显示搜索框提示信息,如果有搜索信息,显示搜索信息. 代码如下: <!DOCTYPE htm ...
- java源文件与类
一个源文件可以包含多个类, 编译的时候,每一个类生成一个字符码文件, 源文件名可以和类名不一致,但字符码文件与类名一致, 如果类是public(公共类),源文件名必须与类名一致 命名规则:源文件的路径 ...
- 【读书笔记】iOS-动态类型和动态绑定
id是泛类型,可以用来存放各种类型的对象,使用id也就是使用“动态类型”. 动态类型,就是指,对象实际使用的是哪一个类是在执行期间确定的,而非在编译期间. 虽然id类型可以定义任何类型的对象,但是不要 ...