为什么ArrayList集合中不能使用foreach增删改?

探讨了在Java中使用ArrayList时,直接在foreach循环中删除元素可能导致的ConcurrentModificationException异常,分析了异常产生的原因,并提供了使用Iterator正确删除元素的方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

来源:http://suo.im/4XaI8Q

编程过程中常常需要使用到集合,而ArrayList也是我们常常使用的,但是最近在一次删除和增加中出现了一些问题,分享记录下。

请看下面两段代码,哪段代码会报错呢,或者都成功呢。

List<String> arrayList1 = new ArrayList<String>();
arrayList1.add("1");
arrayList1.add("2");
for (String s : arrayList1) {
 if("1".equals(s)){
 arrayList1.remove(s);
 }
}
List<String> arrayList2 = new ArrayList<String>();
arrayList2.add("2");
arrayList2.add("1");
for (String s : arrayList2) {
 if("1".equals(s)){
 arrayList2.remove(s);
 }
}

程序运行结果如下:

arrayList1的remove方法成功执行,但是arrayList2的remove方法运行抛出ConcurrentModificationException异常。

为了寻找原因,我们只能看源代码了。

因为foreach的本质就是使用的迭代器Iterator,所有的Collection集合类都会实现Iterable接口。

找到ArrayList类的iterator()方法

使用自己的Itr内部类,并且实现了Iterator接口

迭代器的本质是先调用hasNext()方法判断存不存在下一个元素,然后再使用next()方法取下一个元素

Itr内部类实现

上面arraylist1为什么能remove成功呢,其实它只循环了一次,所以成功了。

因为它在remove元素1之后,它的size - 1变成1,然后Itr内部的cursor变量由0变成1


此时1 = 1,循环结束,所以成功了。

arraylist2为什么remove失败呢,因为他在循环第二次的时候,也remove成功了,但是第三次判断next的时候cursor的值为2导致不等于现在的size 1,所以执行了next方法,最重要的来了,之前remove的操作导致ArrayList的modCount值加1,然后Itr类中的expectedModCount保持不变,所以会抛出异常。

同理可得,由于add操作也会导致modCount自增,所以不允许在foreach中删除、增加、修改ArrayList中的元素。

对此,推荐大家使用迭代器Iterator删除元素,

Iterator<String> iterator = arrayList2.iterator();
while(iterator.hasNext()){
 String item = iterator.next();
 if("1".equals(item)){
 iterator.remove();
 }
}

如果存在并发操作,还需要对Iterator进行加锁操作。

END

Java深入理解JVM - 系统性能优化篇

面试题:说一下Spring 和 Spring Boot 的区别

精选15款Spring开源项目脚手架

很不错!SQL 语法速成,很详细!

SQL别用count查找是否"存在",很耗费时间

关注公众号,查看更多优质文章

有读者问我公众号有没有Java相关的学习资料,我整理了很多Java学习资料和视频放在公众号后台了。


获取方式:关注公众号并回复 Java 领取,更多内容陆续奉上。
点个[在看],支持一下哦~
<think>好的,我需要分析用户提供的Java代码,确定它是否属于增删改功能。首先,用户展示了一个方法getCompanyList,返回类型是CommonResult<List<CompanyInfoVO>>,看起来像是查询数据。方法内部使用了Service的list方法,根据userId查询关联的公司信息,然后转换这些数据到VO对象并返回。整个过程没有出现insert、update或delete操作,而是通过查询获取数据。因此,这段代码属于数据查询功能,而不是增删改。接下来,我需要解释代码的每个部分,说明为什么它不是增删改,并指出可能的问题,比如NPE和性能问题。最后,根据用户的要求,提出相关问题,确保问题与代码中的技术点相关,比如MyBatis-Plus的使用、Lambda查询、VO转换等。</think><< 这段代码实现的是**查询功能**而非增删改功能。以下是关键分析点: 1. **方法签名**: ```java public CommonResult<List<CompanyInfoVO>> getCompanyList(Long userId) ``` - 方法名`getCompanyList`明确表明这是一个查询操作 - 返回类型包含`List<CompanyInfoVO>`说明是数据获取而非修改 2. **数据操作方式**: - 使用`sysUserCompanyService.list()`和`this.list()`进行数据查询 - 全程没有出现`insert/update/delete`等写操作方法 - 最终返回的是通过`CommonResult.success()`包装的查询结果 3. **具体流程**: ```java // 1. 查询用户关联的公司关系记录(SELECT操作) List<SysUserCompany> sysUserCompanyList = sysUserCompanyService.list(查询条件); // 2. 提取公司ID集合 List<Long> companyIds = ...; // 3. 根据公司ID查询公司详细信息(SELECT操作) List<SysCompany> sysCompanyList = this.list(查询条件); // 4. 数据转换(Entity -> VO) List<CompanyInfoVO> companyInfoVOList = ...; // 5. 返回查询结果 return CommonResult.success(...); ``` 4. **技术栈特征**: - 使用MyBatis-Plus的`lambdaQuery()`构建查询条件 - 通过`Wrappers`创建条件构造器 - 涉及DTO/VO的数据转换但无数据修改 **潜在问题**: 1. **空指针风险**:当`userId`为null时,`eq(SysUserCompany::getUserId, userId)`可能引发异常 2. **性能问题**:两次数据库查询(用户-公司关系查询 + 公司详细信息查询)可能存在优化空间 3. **空结果处理**:当查询结果为空时直接返回空列表,可能需要特殊处理逻辑
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值