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

for(Integer i:list){

if(i==3) list.remove(i);

}

System.out.println(list);

抛出异常:java.util.ConcurrentModificationException

foreachList.iterator()源码着手分析,跟踪iterator()方法,该方法返回了 Itr 迭代器对象。

public Iterator iterator() {

return new Itr();

}

Itr 类定义如下:

private class Itr implements Iterator {

int cursor;       // index of next element to return

int lastRet = -1; // index of last element returned; -1 if no such

int 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异常。

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

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

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

Iterator 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 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删除元素时,需要注意索引会左移的问题。

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)

img

最后

关于面试刷题也是有方法可言的,建议最好是按照专题来进行,然后由基础到高级,由浅入深来,效果会更好。当然,这些内容我也全部整理在一份pdf文档内,分成了以下几大专题:

  • Java基础部分

  • 算法与编程

  • 数据库部分

  • 流行的框架与新技术(Spring+SpringCloud+SpringCloudAlibaba)

这份面试文档当然不止这些内容,实际上像JVM、设计模式、ZK、MQ、数据结构等其他部分的面试内容均有涉及,因为文章篇幅,就不全部在这里阐述了。

作为一名程序员,阶段性的学习是必不可少的,而且需要保持一定的持续性,这次在这个阶段内,我对一些重点的知识点进行了系统的复习,一方面巩固了自己的基础,另一方面也提升了自己的知识广度和深度。
《一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码》点击传送门即可获取!
-1711792517742)]

这份面试文档当然不止这些内容,实际上像JVM、设计模式、ZK、MQ、数据结构等其他部分的面试内容均有涉及,因为文章篇幅,就不全部在这里阐述了。

作为一名程序员,阶段性的学习是必不可少的,而且需要保持一定的持续性,这次在这个阶段内,我对一些重点的知识点进行了系统的复习,一方面巩固了自己的基础,另一方面也提升了自己的知识广度和深度。
《一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码》点击传送门即可获取!

  • 22
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: JavaListremove方法是用于从List删除指定元素的方法。该方法有两种重载形式: 1. remove(int index):删除指定索引位置的元素,并返回被删除的元素。 2. remove(Object obj):删除指定元素,如果List存在该元素,则返回true,否则返回false。 例如,以下代码演示了如何使用Listremove方法删除元素: ``` List<String> list = new ArrayList<>(); list.add("apple"); list.add("banana"); list.add("orange"); // 删除指定索引位置的元素 String removedElement = list.remove(1); System.out.println("被删除的元素是:" + removedElement); // 删除指定元素 boolean isRemoved = list.remove("orange"); System.out.println("是否删除成功:" + isRemoved); ``` ### 回答2: 在JavaList是一个非常常用的集合类,它代表一个有序、可重复的集合。在List,可以通过索引访问各个元素,并且支持添加、删除、替换等操作。其remove方法List最常用的方法之一。 Listremove方法有两种重载形式: 1. boolean remove(Object o): 从List删除一个指定元素。如果删除成功,则返回true;否则,返回false。 2. E remove(int index): 从List删除指定索引位置的元素,并返回被删除的元素。 Listremove方法的实现原理如下: 1. 调用remove(Object o)方法时,List遍历整个集合,查找目标元素,并将其删除。 2. 调用remove(int index)方法时,List首先检查索引是否越界,然后将指定索引位置的元素删除,并且将后面的元素向前移动,以填补空缺。 需要注意的是,调用remove(int index)方法时,List的元素索引会发生变化。例如,如果现在有一个List A包含三个元素,分别为a、b、c,它们的索引位置分别为0、1、2。如果我们调用A.remove(1)方法,将b元素删除,那么List A会剩下两个元素:a和c,它们的索引位置分别为0和1,因为现在c的索引位置发生了改变。 在使用Listremove方法时,我们需要注意以下几点: 1. 调用remove(Object o)方法时,要保证目标元素在List存在,否则会返回false。 2. 调用remove(int index)方法时,要确保索引位置的合法性,否则会抛出IndexOutOfBoundsException异常。 3. 在使用Listremove方法时,我们可以使用迭代器来遍历List并删除元素,但要注意不能直接使用Listremove方法,而应该使用迭代器的remove方法来删除元素。 总之,Listremove方法是一个常用的集合操作方法,可以帮助我们方便地删除List的指定元素,提高程序的效率。但在使用时,我们需要注意其具体实现和使用方法。 ### 回答3: List是Java常用的集合类型之一,它提供了一种有序、可重复的集合,可以通过下标访问集合的元素。 Listremove方法用于删除集合指定位置或指定元素的对象。List索引从0开始计数。remove方法的语法如下: ```java boolean remove(Object o) //删除指定元素 E remove(int index) //删除指定位置的元素 ``` 第一种情况,当我们想要删除List集合某个指定的元素时,可以通过调用remove方法并传入要删除元素的值来实现。注意,这里remove方法删除的是传入值相等的元素,并且如果List存在多个相同值的元素,只会删除第一个。 例如: ```java List<Integer> numbers = new ArrayList<Integer>(); numbers.add(1); numbers.add(2); numbers.add(3); numbers.add(2); numbers.remove(Integer.valueOf(2)); //删除第一个值为2的元素 System.out.println(numbers); //[1, 3, 2] ``` 第二种情况,当我们要删除的元素的位置已知时,可以通过调用remove方法并传入要删除元素的下标来实现。注意,这里删除后List所有的元素下标都会自动向前移动一个位置。 例如: ```java List<String> fruits = new ArrayList<String>(); fruits.add("apple"); fruits.add("banana"); fruits.add("orange"); System.out.println("Before remove: " + fruits); //[apple, banana, orange] fruits.remove(1); //删除下标为1(即第二个元素)的元素“banana” System.out.println("After remove: " + fruits); //[apple, orange] ``` 需要注意的是,在使用remove方法时,如果删除的元素是基本类型(如int、char等)时,需要将其封装成对应的对象类型(如Integer、Character等)后再传入remove方法的参数。如果不封装,会被认为是要删除指定的下标位置的元素。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值