点击上方“Java精选”,选择“设为星标”

别问别人为什么,多问自己凭什么!

下方留言必回,有问必答!

每天 08:00 更新文章,每天进步一点点...

Java的List在删除元素时,一般会用list.remove(o)/remove(i)方法。在使用时,容易触碰陷阱,得到意想不到的结果。总结以往经验,记录下来与大家分享。

首先初始化List,代码如下:

package com.cicc.am.test;import java.util.ArrayList;
import java.util.List;public class ListTest {public static void main(String[] args) {List<Integer> list=new ArrayList<Integer>();list.add(1);list.add(2);list.add(3);list.add(3);list.add(4);System.out.println(list);}
}

输出结果为[1, 2, 3, 3, 4]

1、普通for循环遍历List删除指定元素--错误!!!

for(int i=0;i<list.size();i++){if(list.get(i)==3) list.remove(i);
}
System.out.println(list);

输出结果:[1, 2, 3, 4]

为什么元素3只删除了一个?本以为这代码再简单不过,可还是掉入了陷阱里,上面的代码这样写的话,元素3是过滤不完的。只要list中有相邻2个相同的元素,就过滤不完。更多面试题推荐号内,回复Java面试,获取最全面试资料。

List调用remove(index)方法后,会移除index位置上的元素,index之后的元素就全部依次左移,即索引依次-1要保证能操作所有的数据,需要把index-1,否则原来索引为index+1的元素就无法遍历到(因为原来索引为index+1的数据,在执行移除操作后,索引变成index了,如果没有index-1的操作,就不会遍历到该元素,而是遍历该元素的下一个元素)。

如果这样,删除元素后同步调整索引或者倒序遍历删除元素,是否可行呢?

2、for循环遍历List删除元素时,让索引同步调整--正确

for(int i=0;i<list.size();i++){if(list.get(i)==3) list.remove(i--);
}
System.out.println(list);

输出结果:[1, 2, 4]

推荐下自己做的 Spring Boot 的后台管理系统实战项目:

https://gitee.com/yoodb/jing-xuan

3、倒序遍历List删除元素--正确

for(int i=list.size()-1;i>=0;i--){if(list.get(i)==3){list.remove(i);}
}
System.out.println(list);

输出结果:[1, 2, 4]

4、foreach遍历List删除元素--错误!!!

for(Integer i:list){if(i==3) list.remove(i);
}
System.out.println(list);

抛出异常:java.util.ConcurrentModificationException

foreach 写法实际上是对的 Iterable、hasNext、next方法的简写。因此从List.iterator()源码着手分析,跟踪iterator()方法,该方法返回了 Itr 迭代器对象。

public Iterator<E> iterator() {return new Itr();}

Itr 类定义如下:

private class Itr implements Iterator<E> {int cursor;       // index of next element to returnint lastRet = -1; // index of last element returned; -1 if no suchint expectedModCount = modCount;public boolean hasNext() {return cursor != size;}@SuppressWarnings("unchecked")public E next() {checkForComodification();int i = cursor;if (i >= size)throw new NoSuchElementException();Object[] elementData = ArrayList.this.elementData;if (i >= elementData.length)throw new ConcurrentModificationException();cursor = i + 1;return (E) elementData[lastRet = i];}public void remove() {if (lastRet < 0)throw new IllegalStateException();checkForComodification();try {ArrayList.this.remove(lastRet);cursor = lastRet;lastRet = -1;expectedModCount = modCount;} catch (IndexOutOfBoundsException ex) {throw new ConcurrentModificationException();}}final void checkForComodification() {if (modCount != expectedModCount)throw new ConcurrentModificationException();}}

通过代码我们发现 Itr 是 ArrayList 中定义的一个私有内部类,在 next、remove方法中都会调用checkForComodification 方法,该方法的 作用是判断 modCount != expectedModCount是否相等,如果不相等则抛出ConcurrentModificationException异常。

推荐:Java进阶学习资料

每次正常执行 remove 方法后,都会对执行expectedModCount = modCount赋值,保证两个值相等,那么问题基本上已经清晰了,在 foreach 循环中执行 list.remove(item);,对 list 对象的 modCount 值进行了修改,而 list 对象的迭代器的 expectedModCount 值未进行修改,因此抛出了ConcurrentModificationException异常。

5、迭代删除List元素--正确

java中所有的集合对象类型都实现了Iterator接口,遍历时都可以进行迭代:

Iterator<Integer> it=list.iterator();while(it.hasNext()){if(it.next()==3){it.remove();}}
System.out.println(list);

输出结果:[1, 2, 4]

Iterator.remove() 方法会在删除当前迭代对象的同时,会保留原来元素的索引。所以用迭代删除元素是最保险的方法,建议大家使用List过程

中需要删除元素时,使用这种方式。推荐:Java进阶学习资料

6、迭代遍历,用list.remove(i)方法删除元素--错误!!!

Iterator<Integer> it=list.iterator();while(it.hasNext()){Integer value=it.next();if(value==3){list.remove(value);}}
System.out.println(list);

抛出异常:java.util.ConcurrentModificationException,原理同上述方法4.

7、List删除元素时,注意Integer类型和int类型的区别.

上述Integer的list,直接删除元素2,代码如下:

list.remove(2);
System.out.println(list);

输出结果:[1, 2, 3, 4]

可以看出,List删除元素时传入数字时,默认按索引删除。如果需要删除Integer对象,调用remove(object)方法,需要传入Integer类型,代码如下:

list.remove(new Integer(2));
System.out.println(list);

输出结果:[1, 3, 3, 4]

总结

1、用for循环遍历List删除元素时,需要注意索引会左移的问题。

2、List删除元素时,为避免陷阱,建议使用迭代器iterator的remove方式。

3、List删除元素时,默认按索引删除,而不是对象删除。

作者:倚楼听风雨

blog.csdn.net/pelifymeng2/article/details/78085836/

精品资料,超赞福利!

>>Java精选面试题<< - 小程序,3000+ 道面试题在线刷,最新、最全 Java 面试题!

期往精选  点击标题可跳转

【134期】技术总监问:如何设计群聊消息的已读未读功能?

【135期】面试官问:为什么要合并 HTTP 请求,实现batch call?

【136期】面试官:Spring AOP、AspectJ、CGLIB ?它们有什么关系?

【137期】面试官问:RocketMQ 与 Kafka 对比,谈谈两者的差异?

【138期】手撸 websocket + netty 实时视频弹幕交互功能(Java版附源码)

【139期】面试官问:一般后端接口都测试什么?怎么测?

【140期】阿里技术经理问:ReadWriteLock 读写之间互斥吗?

【141期】JDK8 Stream 操作 collectingAndThen:根据对象的属性去重

文章有帮助的话,在看,转发吧!

【142期】List 中 remove() 方法的“陷阱”,被坑惨了!相关推荐

  1. 我被List中remove()方法的陷阱,坑惨了!

    作者:倚楼听风雨 来源:blog.csdn.net/pelifymeng2/article/details/78085836 Java的List在删除元素时,一般会用list.remove(o)/re ...

  2. List中remove()方法的陷阱,被坑惨了!

    欢迎关注方志朋的博客,回复"666"获面试宝典 来源:blog.csdn.net/pelifymeng2/ article/details/78085836 Java的List在删 ...

  3. List中remove()方法的陷阱以及终极的解决方法!

    Java的List在删除元素时,一般会用list.remove(o)方法.在使用时,可能会抛出java.util.ConcurrentModificationException的异常.总结以往经验,记 ...

  4. List中remove()方法的陷阱,开发谨记!

    作者:倚楼听风雨 来源:blog.csdn.net/pelifymeng2/article/details/78085836 Java的List在删除元素时,一般会用list.remove(o)/re ...

  5. List集合中remove方法的使用

    以ArrayList为例: //泛型为引用数据类型: ArrayList<String> list=new ArrayList<String>();list.add(" ...

  6. jquery中remove()方法移除被选元素,包括所有的文本和子节点

    jquery中remove方法移除被选元素,包括所有的文本和子节点 定义和用法 移除所有的 元素 移除html中指定id的元素内容 定义和用法 remove() 方法移除被选元素,包括所有的文本和子节 ...

  7. js中toFixed方法的两个坑

    js中toFixed方法的两个坑 toFixed返回结果是string,后续使用它计算会错误 toFixed返回结果可能出现负零-0.00 toFixed返回结果是string,后续使用它计算会错误 ...

  8. List中remove()方法的注意事项

    集合中remove注意事项 错误使用: 1.普通for循环遍历List删除指定元素错误 for(int i=0;i<list.size();i++){if(list.get(i)==3) {li ...

  9. ArrayList中remove()方法删除元素之后下标重定位的问题

    需求: 有一个ArrayList数组,要求删除长度大于5的字符串,如:arr = {"ab1","123ad","bca","da ...

最新文章

  1. python 聚类_使用python+sklearn实现聚类性能评估中随机分配对聚类度量值的影响
  2. 如何从网页上下载Flash?
  3. 到底该不该从开发转测试
  4. 【已解决】window10任务栏图标显示异常解决方法--有详细解释
  5. 【Linux导论】通过图形接口配置系统(System Configuration from the Graphical Interface)
  6. 强行杀windows服务
  7. 在web开发中,如何保证对象只被创建一次之单例应用系列?
  8. lodash 源码解读 _.findIndex(obj_array, fn)
  9. java循环1000000000_Java11比Java8快多少,不是测试人员一眼也能看懂
  10. 数据分析如何揭示冠状病毒的真相?
  11. 每日算法系列【LeetCode 315】计算右侧小于当前元素的个数
  12. 最优矩阵链乘(动态规划)
  13. vue项目基本环境的配置与初始化
  14. 华为hcip认证考试题库有哪些内容?华为hcip认证考试题库试题举例
  15. 一个函数搞定无限层级分类
  16. word论文排版插件_【Office Word】论文排版有关技巧
  17. java中Date类之GMT、UTC
  18. springcloud之服务发现笔记
  19. 学习 spring-cloud-aibaba第七篇,JWT认证授权
  20. C++大小写转换tolower/toupper以及transform函数

热门文章

  1. 星星之火-28:什么是CDMA的远近效应与功率控制?
  2. sql 基础语法 创建数据库和数据表 数据增删改查 分组查询 子查询回顾
  3. H5聊天对话气泡的一种实现方式及原理
  4. **RAKsmart服务器租用相对其他美国服务器的优势**
  5. 搜索百度网盘资源的方法
  6. STM32 PWM控制舵机
  7. Incapsula到底是什么
  8. 社招简历老被拒?大佬告诉你面试官只中意这些简历,快来看看吧!
  9. dell g7 重装win10
  10. 联想笔记本屏幕扩展快捷键用不了