摘要:介绍List集合实现元素边遍历边删除的方法,例如removeIf和迭代器iterator.remove()等。

综述

  List集合是我们开发中经常使用到的一种集合形式,有时候会遇到在遍历List集合时需要删除指定的元素。但在根据条件使用for循环或者增强的for循环遍历删除某些元素时却不能随心所欲地使用remove函数,我们今天便从实现层面讨论下原因以及Iterator的相关知识。伸手党或者资深大佬看完下面的总结就行,不必阅读全文。

  比如,有一个字符串类型的List,包含四个元素:“软件”、“del”、“开发”、“领路人”,我们要实现的是在遍历这个集合时,删除长度为2的字符串元素。要怎么做呢?有如下几种方法:

  • for循环倒序删除
  • iterator遍历删除
  • 使用removeIf删除

for循环倒序删除

顺序删除时,会报错,提示:Exception in thread "main" java.util.ConcurrentModificationException

for(int i=0;i < list.size();i++){
    if(list.get(i).equals("del"))
        list.remove(i);
}

  这种方式的问题在于,删除某个元素后,list的大小发生了变化,而数组下标也在变化,所以会导致在遍历的时候漏掉某些元素。比如删除第1个元素后,继续根据角标访问第2个元素时,因为删除的关系后面的元素都往前移动了一位,所以实际访问的是第3个元素。因此,这种方式可以用在删除特定的一个元素后就终止循环时使用,尤其不适合循环删除多个元素时使用。

结论:for循环正序删除的问题在于,删除某个元素后,其后元素的数组下标将向前挪一位,所以会导致在遍历的时候漏掉被删除元素的后一个元素。解决办法:从list最后一个元素开始从后向前遍历。

    /**
* 倒序删除:删除list集合中属性全为null的对象
*
*/
private void removeEmptyObjectInList(List<? extends Object> list) {
if (null != list && list.size() > 0) {
for (int i = list.size() - 1; i >= 0; i--) {
if(list.get(i) == null){
list.remove(i);
}
}
}
} public static void main(String[] args) {
List<String> list = new LinkedList<String>();
list.add("a");
list.add("b");
list.add("c");
list.add("d");
list.add("e");
for(int i=0;i<list.size();i++){
System.out.println("即将删除的元素:"+list.get(i));
list.remove(i);
i--;
}
}

当要针对角标进行元素的remove时,使用倒序遍历的方式最为妥当。

使用foreach删除

使用增强型for循环 foreach 删除指定元素:

for(String x:list){
    if(x.equals("del")){
System.out.println("即将删除的元素:"+x);
        list.remove(x);
}
}

  这种方式的问题在于,删除元素后继续循环会抛异常ConcurrentModificationException,因为元素在使用的时候发生了并发修改,导致异常被抛出。但是删除完毕马上使用break跳出,则不会触发报错。

iterator遍历删除

  下面提供一个用迭代器在遍历List集合时删除元素的方法:

Iterator it = list.iterator();
while(it.hasNext()){
    String x = it.next();
    if(x.equals("del")){
System.out.println("即将删除的元素:"+x);
        it.remove();
    }
}

  能达到预期效果吗?我们执行后验证一下就知道了。在控制台可以看到如下结果,说明已经被删除:

即将删除的元素:del

  这种方式可以正常的循环及删除。但要注意的是,调用iterator的remove方法时,如果用list的remove方法同样会报上面提到的ConcurrentModificationException错误。

使用removeIf删除

  在 Java 8 中,Collection及其子类新加入了removeIf函数,作用是按照特定规则过滤集合中的元素。语法为:

arraylist.removeIf(Predicate<E> filter)

  参数说明:filter - 过滤器,判断元素是否被删除,如果元素被删除则filter返回 true。实例,删除等于del的元素:

list.removeIf(s-> s.equals("del"));

结束语

  总结:

  (1)删除list中某一个元素时,可以使用上述几种方式中的任意一种。

  (2)循环删除list中多个元素时,应该使用for循环倒序删除、迭代器iterator或者removeIf

  以上就是小编分享给各位程序猿的全部内容了,希望各位读后有所收获,也希望大家多多支持楼兰胡杨。

JAVA 循环删除list中元素的方法总结的更多相关文章

  1. JAVA中循环删除list中元素的方法总结【转】

    印象中循环删除list中的元素使用for循环的方式是有问题的,但是可以使用增强的for循环,然后今天在使用时发现报错了,然后去科普了一下,再然后发现这是一个误区.下面就来讲一讲..伸手党可直接跳至文末 ...

  2. java中循环删除list中元素的方法

    重点哈 印象中循环删除list中的元素使用for循环的方式是有问题的,但是可以使用增强的for循环,然后今天在使用时发现报错了,然后去科普了一下,再然后发现这是一个误区.下面就来讲一讲..伸手党可直接 ...

  3. JS实现循环删除数组中元素的方法介绍

    这篇文章主要给大家介绍了关于Javascript循环删除数组中元素的几种方法,文中给出了详细的示例代码供大家参考学习,对大家具有一定的参考学习价值,需要的朋友们下面来一起看看吧. 本文主要跟大家分享了 ...

  4. Java中循环删除list中元素的方法总结

    印象中循环删除list中的元素使用for循环的方式是有问题的,但是可以使用增强的for循环,然后在今天使用的时候发现报错了,然后去科普了一下,发现这是一个误区.下面我们来一起看一下. Java中循环遍 ...

  5. JAVA中循环删除list中元素的方法总结(同上篇)

    印象中循环删除list中的元素使用for循环的方式是有问题的,但是可以使用增强的for循环,然后今天在使用时发现报错了,然后去科普了一下,再然后发现这是一个误区.下面就来讲一讲..伸手党可直接跳至文末 ...

  6. python中循环删除列表中元素时的坑!

    循环删除列表中元素时千万别用正序遍历,一定要用反序遍历! 废话不多说,先上案例代码: def test(data): for i in data: data.remove(i) return data ...

  7. JAVA中循环删除list中元素

    文章来源: https://www.cnblogs.com/pcheng/p/5336903.html JAVA中循环遍历list有三种方式for循环.增强for循环(也就是常说的foreach循环) ...

  8. js删除数组中元素的方法

    一.清空数组 var ary = [1,2,3,4]; ary.splice(0,ary.length);//清空数组 console.log(ary); // 输出 [],空数组,即被清空了 二.删 ...

  9. python删除列表中元素的方法

    删除列表中元素的三种方法-remove.pop.del 1 1.remove: 删除单个元素,删除首个符合条件的元素,按值删除 2 举例说明: 3 >>> str=[1,2,3,4, ...

  10. Javascript循环删除数组中元素的几种方法示例

    发现问题 大家在码代码的过程中,经常会遇到在循环中移除指定元素的需求.按照常规的思路,直接一个for循环,然后在循环里面来个if判断,在判断中删除掉指定元素即可.但是实际情况往往不会像预想的那样顺利运 ...

随机推荐

  1. Vue3路由进阶实战:深度解析参数传递与导航守卫核心技术

    一.路由参数传递的进阶应用技巧 1.1 路由配置与参数验证 // router/index.js { path: '/user/:userId(\\d+)', // 使用正则表达式限制只匹配数字 na ...

  2. 大模型知识引擎 LKE 新手入门指南:官方文档难懂?看这篇就够了

    昨天简单体验了一下大模型知识引擎(LKE),总体来说,虽然其功能方面还有一定欠缺,但在一些特定领域,特别是RAG(Retrieval-Augmented Generation)技术的应用上,还是表现出 ...

  3. Golang 入门 : 浮点数

    浮点数介绍 Go语言提供了两种精度的浮点数:float32 和 float64.它们的算术规范由IEEE754浮点数国际标准定义,该浮点数规范被所有现代的CPU支持. 这些浮点数类型的范围可以从很微小 ...

  4. 防止恶意解析——禁止通过IP直接访问网站

    一.什么是恶意解析 一般情况下,要使域名能访问到网站需要两步,第一步,将域名解析到网站所在的主机,第二步,在web服务器中将域名与相应的网站绑定.但是,如果通过主机IP能直接访问某网站,那么把域名解析 ...

  5. JavaScript将时间戳转化为时间

    const timestampToTime= (timestamp) => { const date = new Date(timestamp * 1000); const year = dat ...

  6. 【Docker】安装

    Docker安装 前提说明 CentOS Docker 安装 Docker支持以下的CentOS版本: CentOS 7 (64-bit) CentOS 6.5 (64-bit) 或更高的版本 前提条 ...

  7. CSS文本超出省略

    语法: text-overflow:clip|ellipsis|"任意字符" <!DOCTYPE html> <html> <head> < ...

  8. nbhh的泛型:TDictionary

    type TCity = class Country: String; Latitude: Double; Longitude: Double; end; const EPSILON = 0.0000 ...

  9. 我要成为node_modules大师!(一):包管理器选择,依赖关系分析

    好家伙 1.npm曾经的一些问题 1. 依赖地狱(Dependency Hell) 嵌套依赖结构:早期版本的 npm 采用嵌套的 node_modules 结构,依赖层级极深,容易导致路径过长问题(尤 ...

  10. 谷歌SRE的7条原则

    谷歌SRE的7条原则 拥抱合理的风险 最大化系统的稳定性不仅毫无意义,而且会适得其反.不切实际的可靠性目标限制了新功能交付给用户的速度,而且用户通常不会注意到极端的可用性(比如99.99999%),因 ...