关于报错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);

{

		//查询结构内     基本数据及  联合数据 ,返回统计结果     拿到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)


{

		//查询结构内     基本数据及  联合数据 ,返回统计结果     拿到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按钮就会变绿变亮

在这里插入图片描述

问题触发代码:↓↓↓
在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值