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

Java之遍历List集合安全地删除元素

Java之遍历List集合安全地删除元素

在Java中,遍历一个List并安全地删除元素是一个需要注意的问题。因为直接在遍历过程中修改集合(如删除元素)可能会导致ConcurrentModificationException异常。这是因为集合的迭代器在检测到集合在迭代过程中被修改时,会抛出这个异常来防止不确定的行为。

为了安全地遍历并删除List中的元素,你可以使用以下几种方法:

方法一:使用Iterator

使用Iterator来遍历集合,并通过Iterator.remove()方法来删除元素。这是因为在Iterator的实现中,删除操作是与迭代状态同步的,所以不会引发ConcurrentModificationException。

List<String> list = new ArrayList<>();
list.add("a");
list.add("b");
list.add("c");Iterator<String> iterator = list.iterator();
while (iterator.hasNext()) {String item = iterator.next();if (item.equals("b")) {iterator.remove(); // 安全删除}
}

方法二:使用Java 8的removeIf(如果可用)

如前所述,removeIf方法接受一个Predicate,并安全地删除所有满足条件的元素。这是因为它在内部使用了迭代器的逻辑来避免并发修改异常。

List<String> list = new ArrayList<>();
list.add("a");
list.add("b");
list.add("c");list.removeIf(item -> item.equals("b")); // 安全删除

使用Java 8的流(Streams)‌

虽然流主要用于处理集合的转换和聚合操作,但你可以通过创建一个新的集合来间接地删除元素。这不是原地修改集合,而是创建了一个新的集合。

List<String> list = new ArrayList<>();
list.add("a");
list.add("b");
list.add("c");List<String> filteredList = list.stream().filter(item -> !item.equals("b")).collect(Collectors.toList());// 如果需要,可以替换原集合
list = filteredList;

使用CopyOnWriteArrayList‌

如果你需要一个线程安全的List,并且希望在遍历时能够修改集合,可以使用CopyOnWriteArrayList。这个类在每次修改时都会创建一个新的集合副本,所以迭代时不会受到并发修改的影响。但是,这种方法有很高的开销,因为每次修改都需要复制整个集合。

List<String> list = new CopyOnWriteArrayList<>();
list.add("a");
list.add("b");
list.add("c");for (String item : list) {if (item.equals("b")) {list.remove(item); // 在CopyOnWriteArrayList中是安全的,但性能较差}
}

注意:尽管CopyOnWriteArrayList在迭代时允许修改,但由于其每次修改都会复制整个集合,所以在大量修改操作时性能会非常差。

在大多数情况下,使用Iterator或removeIf方法是最合适的选择,因为它们既安全又高效。如果你需要在遍历时进行复杂的逻辑判断或处理,Iterator可能更灵活一些。而如果你只是简单地根据条件删除元素,removeIf则更加简洁和直观。

http://www.lryc.cn/news/485169.html

相关文章:

  • ceph的集群管理
  • STM32 设计的较为复杂的物联网项目,包括智能家居控制系统,涵盖了硬件和软件的详细设计。
  • Kettle配置数据源错误“Driver class ‘org.gjt.mm.mysql.Driver‘ could not be found”解决记录
  • 二分搜索的三种方法
  • 使用python编写工具:快速生成chrome插件相关文件结构
  • 内存、显存和GPU在Transformer架构中承担什么计算任务
  • 【计算机网络】TCP协议特点3
  • 移植LVGL8.2以及移植过程的理解
  • 动态规划-背包问题——1049.最后一块石头的重量II
  • 【C++学习(37)】并发性模式:如生产者-消费者、读写锁等。 架构模式:如MVC、MVVM等。属于23 种设计模式吗? RAII 的关系?
  • [Mysql] Mysql的多表查询----多表关系(下)
  • 命名空间(namespace)详解(一)
  • HarmonyOS ArkTs 解决流式传输编码问题
  • NPOI 实现Excel模板导出
  • 【OpenGL】OpenGL简介
  • shell命令笔记
  • qml显示OpenCV mat图片
  • 类与对象(2)---类的6个默认成员函数
  • 华为云租户网络-用的是隧道技术
  • 手搓神经网络(MLP)解决MNIST手写数字识别问题 | 数学推导+代码实现 | 仅用numpy,tensor和torch基本计算 | 含正反向传播数学推导
  • esp32c3安装micropython环境
  • ES6的Iterator 和 for...of 循环
  • 《C语言程序设计现代方法》note-4 基本类型 强制类型转换 类型定义
  • MySQL(4)【数据类型 —— 数值类型】
  • Golang超详细入门教程
  • 鸿蒙NEXT自定义组件:太极Loading
  • FPGA 第7讲 简单组合逻辑译码器
  • opencv kdtree pcl kdtree 效率对比
  • 1+X应急响应(网络)系统备份:
  • python os.path.dirname(path) 详解