关于报错java.util.ConcurrentModificationException: null的源码分析和解决

一般有这种问题,方法中至少会有List或者Map下的至少两个子类,有可能参数类型相同,也有可能不同都有可能触发这个问题!其主要原因是使用了ArrayList进行删除操作或者使用iterator遍历集合的同时对集合进行修改都有可能会出现这个问题

ArrayList属于List下的子类

需要区分的是List在java中有两个属于两个不同的包,这里说的是Util包下的List!两个类一个是接口一个是Class类(基础知识了吧算是)

针对此问题的ArrayList源码解析https://blog.csdn.net/qq_43705131/article/details/122607384

问题截图及源码触发部分代码

业务问题代码

不管泛型使用的是同种参数还是不同参数都要注意这个问题

场景复现

业务场景:

需要一个接口返回一个List对象,但是在List中需要调用另一个接口并将该接口的返回结果插入List中(如上图代码注释所写)

这时候类型的兼容性问题就出现了

代码场景:

首先:调用接口返回一个集合,需要集合中的全部数据及拿到集合中的ID

List<Map<String, List>> periodicLQ = periodicObservationMapper.periodicLQ(pointId,dir, lineNo, startDate, endDate);

用于调用另一个接口查询该ID下的全部值并且赋到一个List中用于返回

List structureDetailInfo = structureDetailInfoMapper.selectByPavementStructureInfoIds(ids);

代码实现:(问题复现)

这样写在List structureDetailInfo = structureDetailInfoMapper.selectByPavementStructureInfoIds(ids);

java 复制代码
{

		//查询结构内     基本数据及  联合数据 ,返回统计结果     拿到structure_info 的 ID拿到结构数据将此以数组返回到VO
		List<Map<String, List<StructureDetailInfoPO>>> periodicLQ = periodicObservationMapper.periodicLQ(pointId,dir, lineNo, startDate, endDate);

		Map<String, List<StructureDetailInfoPO>> structureInfo = new HashMap<>();

		//用于存储接口获取到的所有id值
		List<Long> ids = new ArrayList<>();
		for (Map<String, List<StructureDetailInfoPO>> item: periodicLQ) {

			if (!item.isEmpty()){
			Long id = Long.valueOf(String.valueOf(item.get("id")));
			ids.add(id);
				for (int i = 0;i <= item.size();i++){

			
					List<StructureDetailInfoPO> structureDetailInfo = structureDetailInfoMapper.selectByPavementStructureInfoIds(ids);
					structureInfo.put("structureInfo",structureDetailInfo);
					//System.out.println(" structure detail info size : " + structureDetailInfo);
					
					//resultMap.add(structureInfo);
				}
					result.add(item);
					//result.add(structureInfo);
					//break;
			}
			System.out.println(" result size : " + result);
		}

		return result;
	}

解决方案-->避免修改ArrayList或者引用的其他List类和Map中的子类

创建一个名为 resultMap 的临时列表,并在循环中将结构信息添加到该列表中。最后,我们返回 resultMap 列表,而不是修改 result 列表用于返回。这就能够解决 ConcurrentModificationException 异常的问题,因为这样就创建了一个中间变量用来存储结果,避免修改了原来的result列表(List)

java 复制代码
{

		//查询结构内     基本数据及  联合数据 ,返回统计结果     拿到structure_info 的 ID拿到结构数据将此以数组返回到VO
		List<Map<String, List<StructureDetailInfoPO>>> periodicLQ = periodicObservationMapper.periodicLQ(pointId,dir, lineNo, startDate, endDate);

		List<Map<String, List<StructureDetailInfoPO>>> resultMap = new ArrayList<>();
		Map<String, List<StructureDetailInfoPO>> structureInfo = new HashMap<>();

		//用于存储接口获取到的所有id值
		List<Long> ids = new ArrayList<>();
		for (Map<String, List<StructureDetailInfoPO>> item: periodicLQ) {

			if (!item.isEmpty()){
			Long id = Long.valueOf(String.valueOf(item.get("id")));
			ids.add(id);
				for (int i = 0;i <= item.size();i++){

					if (i == item.size()){

					List<StructureDetailInfoPO> structureDetailInfo = structureDetailInfoMapper.selectByPavementStructureInfoIds(ids);
					structureInfo.put("structureInfo",structureDetailInfo);
					//System.out.println(" structure detail info size : " + structureDetailInfo);
					}else {
					    continue;
					}

					//resultMap.add(structureInfo);
				}
					resultMap.add(item);
					resultMap.add(structureInfo);
					//break;
			}
			System.out.println(" result size : " + resultMap);
		}

		return resultMap;
	}

:w最后总结:

代码中有

Long id = Long.valueOf(String.valueOf(item.get("id")));

在写这个的时候,一开始用了toString,编译不报错但运行有错,也建议大家使用这种包围的方法取代.***的方法(这种问题一定会随着jdk的不断更新有所改正)

此外在对断点进行一步步跟踪到源代码时发现的问题触代码如下

如图所示我们在读过SQLsession等常规源码后发现代码执行到了此类中,并确认了SQL是否执行正确

在左下角即为控制程序执行的断点按钮,可以手动点击||,这样resume按钮就会变绿变亮

问题触发代码:↓↓↓

相关推荐
shandianchengzi43 分钟前
【科普】安卓|安卓手机上如何简便实现Ctrl+Z(需要键盘或一台Windows电脑)
android·windows·智能手机·计算机外设·安卓·科普·记录
浮尘笔记2 小时前
Java Snowy框架CI/CD云效自动化部署流程
java·运维·服务器·阿里云·ci/cd·自动化
kkeeper~9 小时前
0基础C语言积跬步之深入理解指针(5下)
c语言·开发语言
一直不明飞行9 小时前
Java的equals(),hashCode()应该在什么时候重写
java·开发语言·jvm
REDcker9 小时前
有限状态机与状态模式详解 FSM建模Java状态模式与C++表驱动模板实践
java·c++·状态模式
盲敲代码的阿豪9 小时前
Python 入门基础教程(爬虫前置版)
开发语言·爬虫·python
你的保护色9 小时前
【无标题】
java·服务器·网络
basketball61610 小时前
C++ 构造函数完全指南:从入门到进阶
java·开发语言·c++
互联科技报10 小时前
2026超融合选型:Top5品牌与市场格局解读
开发语言·perl
weixin1997010801610 小时前
[特殊字符] 智能数据采集:数字化转型的“数据石油勘探队”(附Python实战源码)
开发语言·python