当前位置: 首页 > article >正文

任何时候都不要在 for 循环中删除 List 集合元素!!!

首先说结论:无论什么场景,都不要对List使用for循环的同时,删除List集合元素,因为这么做就是不对的。

阿里开发手册也明确说明禁止使用foreach删除、增加List元素。
正确删除元素的方式是使用迭代器(Iterator),代码如下:

List<String> list = new ArrayList<>();
Iterator<String> iterator = list.iterator();
while (iterator.hasNext()) {
    // 删除元素
    iterator.remove();
}

JDK8后lambda写法:

list.removeIf(s -> s.contains("a"));

不想知道为什么不能使用for循环删除List集合元素的,看完前言就可以关闭本页面了,想知道原因的继续往下看

下面举个实例场景,看一下为什么不能使用for循环。

需求
一个List集合,元素类型为String,有N个元素,删除这些元素中包含字符’‘a’'的元素。

假设集合内容如下:

List<String> list = new ArrayList<>(4);
list.add("a");
list.add("ab");
list.add("abc");
list.add("abcd");
public static void main(String[] args) {
    List<String> list = new ArrayList<>(4);
    list.add("a");
    list.add("ab");
    list.add("abc");
    list.add("abcd");

    Iterator<String> iterator = list.iterator();
    while (iterator.hasNext()) {
        if (iterator.next().contains("a")) {
            // 删除元素
            iterator.remove();
        }
    }
    System.out.println(list);
}

输出结果为

[]

错误答案1:普通for循环(for-i)

public static void main(String[] args) {
    List<String> list = new ArrayList<>(4);
    list.add("a");
    list.add("ab");
    list.add("abc");
    list.add("abcd");

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

输出结果为

[ab, abcd]

分析
普通for循环遍历List集合的同时,删除List中的元素是可以运行的代码,但在大多数场景下,不能使用这种方式,上边的结果也印证了这一点,虽然你的代码不会报错,运行也正常,但在本实例中,这么写就是BUG。

BUG原因:索引为i的元素删除后,后边元素的索引自动向前补位,即原来索引为i+1的元素,变为了索引为i的元素,但是下一次循环取的索引是i+1,此时你以为取到的是原来索引为i+1的元素,其实取到是原来索引为i+2的元素。

只要每删除一个元素,就会漏掉下一个元素,所以这种方式从逻辑上来说是存在bug的,无论什么需求场景,都不建议用这种方式,因为不可控因素太多(鬼知道生产环境中他会删掉多少元素,同时漏掉多少元素)。

既然这么写不报错,那么个别特殊场景确实可以使用这种普通for循环删除元素的,比如我们把实例的需求变动一下,改为:一个List集合,元素类型为String,有N个元素,删除这些元素中包含字符’a’的元素,如果有连续两个或以上元素包含’a’,那么只删除当前连续元素中的奇数位元素。虽然这种场景适用,但仍然不推荐,还是因为太不可控。

错误答案2:增强for循环(foreach)

public static void main(String[] args) {
    List<String> list = new ArrayList<>(4);
    list.add("a");
    list.add("ab");
    list.add("abc");
    list.add("abcd");

    for (String str : list) {
        if (str.contains("a")) {
            list.remove(str);
        }
    }
    System.out.println(list);
}

运行报错:

Exception in thread "main" java.util.ConcurrentModificationException
 at java.util.ArrayList$Itr.checkForComodification(ArrayList.java:909)
 at java.util.ArrayList$Itr.next(ArrayList.java:859)
 at top.oldmoon.learn.test.ListTest.main(ListTest.java:24)

使用百度翻译可以知道:Concurrent Modification Exception:并发修改异常

分析

可以简单的理解为:foreach就不支持对集合中的元素进行增删操作,但是可以修改。


http://www.kler.cn/a/301.html

相关文章:

  • C#Halcon图像处理畸变校正之曲面校正
  • HarmonyOS开发:关于帧动画使用分享
  • SimForge HSF 案例分享|复杂仿真应用定制——UAVSim无人机仿真APP(技术篇)
  • 基于Redis有序集合实现滑动窗口限流
  • 利用JavaScript实现猜数字
  • day-102 二进制矩阵中的最短路径
  • 为什么软件测试面试了几个月都没有offer,从HR角度分析
  • 2月编程语言排行榜新鲜出炉,谁又摘得桂冠?
  • 第 46 届世界技能大赛浙江省选拔赛“网络安全“项目C模块任务书
  • Verilog实现组合逻辑电路
  • CANoe中使用CAPL函数接口调用Vflash文件
  • 【面试题】面试官:如果后端给你 1w 条数据,你如何做展示?
  • 【前端老赵的CSS简明教程】10-1 CSS预处理器和使用方法
  • 学习C++这几个网站足矣
  • 如何将项目部署到服务器:从选择服务器到维护应用程序的全流程指南
  • 【Java实战】不会还有人用if else进行参数校验吧
  • linux进程管理
  • 代码规范(C++)
  • 【拳打蓝桥杯】最基础的数组你真的掌握了吗?
  • 利用Postman的简单运用解决小问题的过程
  • 2023年中国高校计算机大赛-团队程序设计天梯赛(GPLT)上海理工大学校内选拔赛(同步赛) A — E
  • 蓝桥杯刷题第十天
  • 前端安全:如何保障 Web 应用程序的安全性?
  • leetcode刷题 | 关于前缀树的题型总结
  • 世界顶级五大女程序媛,不仅技术强还都是美女
  • 第十二届蓝桥杯省赛详解