关于报错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按钮就会变绿变亮

问题触发代码:↓↓↓

相关推荐
csbysj20202 分钟前
DTD 元素:XML 与 SGML 文档结构解析指南
开发语言
傻童:CPU16 分钟前
C语言练习题
c语言·开发语言
华仔啊24 分钟前
JVM参数到底配在哪?7大场景全解,新手不再迷茫!
java·jvm
极地星光26 分钟前
协程:实战与系统集成(高级篇)
开发语言
0和1的舞者1 小时前
《Git:从入门到精通(八)——企业级git开发相关内容》
大数据·开发语言·git·搜索引擎·全文检索·软件工程·初学者
盒马盒马1 小时前
Rust:Windows 系统 VsCode 环境搭建
windows·vscode·rust
liulilittle1 小时前
LwIP协议栈MPA多进程架构
服务器·开发语言·网络·c++·架构·lwip·通信
水淹萌龙1 小时前
玩转 Go 表达式引擎:expr 实战指南
开发语言·后端·golang
艾莉丝努力练剑1 小时前
【C++:继承】面向对象编程精要:C++继承机制深度解析与最佳实践
开发语言·c++·人工智能·继承·c++进阶
penguin_bark1 小时前
C++ 异步编程(future、promise、packaged_task、async)
java·开发语言·c++