0

0

什么是集合的结构化修改_add/remove对迭代器状态的影响说明

P粉602998670

P粉602998670

发布时间:2026-02-24 18:42:10

|

768人浏览过

|

来源于php中文网

原创

concurrentmodificationexception 是因迭代器检测到集合被结构性修改而触发的异常;它通过比对 modcount 和 expectedmodcount 实现,仅 iterator.remove() 能安全同步状态,其他 add/remove/clear 等操作均会引发该异常。

什么是集合的结构化修改_add/remove对迭代器状态的影响说明

ConcurrentModificationException 是怎么被触发的

当你在用 Iterator 遍历集合时,调用 c.remove(obj)c.add(obj),几乎一定会抛出 ConcurrentModificationException。这不是“运气不好”,而是 Java 故意设计的保护机制——它通过维护一个叫 modCount(修改计数)的字段,和迭代器内部的 expectedModCount 做比对,一旦不一致就立刻报错。

  • modCount 在每次调用 add/remove/clear 等结构性方法时自增
  • Iterator 构造时会把当前 modCount 复制给自己的 expectedModCount
  • 每次调用 next()hasNext() 前,都会执行 checkForComodification() 检查两者是否相等
  • 所以哪怕你只删了一个元素,后续再调一次 next() 就崩

为什么 Iterator.remove() 是唯一安全的删除方式

Iterator.remove() 不是“绕过检查”,而是主动同步状态:它在调用集合底层 remove() 的同时,把 expectedModCount 重新设为当前 modCount,让下一次检查能通过。

  • 它只能删掉上一次 next() 返回的那个元素(即 lastRet 位置),不能随意删任意值
  • 必须在 next() 之后、且未被删过的情况下调用,否则抛 IllegalStateException
  • 不能在 for-each 循环里用——因为语法糖隐藏了 Iterator,你根本拿不到引用
Iterator<String> it = c.iterator();
while (it.hasNext()) {
    String str = it.next();
    if ("#".equals(str)) {
        it.remove(); // ✅ 安全
        // c.remove(str); // ❌ 立刻触发 CME
    }
}

哪些操作算“结构性修改”

不是所有修改都会触发异常。“结构性修改”特指改变集合大小或物理存储结构的操作,和“改内容”有本质区别:

梯子AI
梯子AI

百度推出的AI智能搜索

下载
  • 会触发 CME:add()remove()clear()retainAll()removeAll()
  • 不会触发 CME:set()(仅替换某索引处元素)、get()contains()size()
  • 注意:ArrayList.set(i, x) 安全;但 LinkedList.set(i, x) 内部可能涉及节点调整,某些 JDK 版本仍会更新 modCount,建议统一按“可能触发”处理

想边遍历边增删?别硬刚,换思路

硬要在迭代中增删,本质上是在对抗迭代器的设计契约。与其反复踩坑,不如选更匹配的方案:

  • 要过滤元素(如删掉所有 "#")→ 收集待删项,遍历完再批量 removeAll()
  • 要构建新集合 → 用 Stream.filter() 或新建 ArrayList + add()
  • 高并发读多写少 → 考虑 CopyOnWriteArrayList,但注意它的迭代器看到的是快照,删完也看不到效果
  • 真需要边走边改且单线程 → 用传统 for (int i = 0; i ,删元素时手动 <code>i-- 补位

最常被忽略的一点:即使没报异常,用集合方法删元素也可能导致漏删或重复遍历——因为底层数组位移后,迭代器游标还按原节奏走。这比崩溃更危险,它悄悄错得离谱。

相关标签:

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

850

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

581

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

294

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

210

2025.08.29

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

719

2023.08.10

Golang 生态工具与框架:扩展开发能力
Golang 生态工具与框架:扩展开发能力

《Golang 生态工具与框架》系统梳理 Go 语言在实际工程中的主流工具链与框架选型思路,涵盖 Web 框架、RPC 通信、依赖管理、测试工具、代码生成与项目结构设计等内容。通过真实项目场景解析不同工具的适用边界与组合方式,帮助开发者构建高效、可维护的 Go 工程体系,并提升团队协作与交付效率。

1

2026.02.24

Golang 性能优化专题:提升应用效率
Golang 性能优化专题:提升应用效率

《Golang 性能优化专题》聚焦 Go 应用在高并发与大规模服务中的性能问题,从 profiling、内存分配、Goroutine 调度、GC 机制到 I/O 与锁竞争逐层分析。结合真实案例讲解定位瓶颈的方法与优化策略,帮助开发者建立系统化性能调优思维,在保证代码可维护性的同时显著提升服务吞吐与稳定性。

2

2026.02.24

Golang 面试题精选:高频问题与解答
Golang 面试题精选:高频问题与解答

Golang 面试题精选》系统整理企业常见 Go 技术面试问题,覆盖语言基础、并发模型、内存与调度机制、网络编程、工程实践与性能优化等核心知识点。每道题不仅给出答案,还拆解背后的设计原理与考察思路,帮助读者建立完整知识结构,在面试与实际开发中都能更从容应对复杂问题。

1

2026.02.24

Golang 运行与部署实战:从本地到云端
Golang 运行与部署实战:从本地到云端

《Golang 运行与部署实战》围绕 Go 应用从开发完成到稳定上线的完整流程展开,系统讲解编译构建、环境配置、日志与配置管理、容器化部署以及常见运维问题处理。结合真实项目场景,拆解自动化构建与持续部署思路,帮助开发者建立可靠的发布流程,提升服务稳定性与可维护性。

3

2026.02.24

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号