解决JAVA8 Collectors.toMap value为null报错的问题
2018年11月7日17:59:27该bug貌似在java9中修复,欢迎补充
2019年3月19日17:59:11查看java11的toMap方法后,发现并没有修改任何实现
Causedby:java.lang.NullPointerException java.util.HashMap.merge(HashMap.java:1224) java.util.stream.Collectors.lambda$toMap$58(Collectors.java:1320) java.util.stream.ReduceOps$3ReducingSink.accept(ReduceOps.java:169) java.util.ArrayList$ArrayListSpliterator.forEachRemaining(ArrayList.java:1374) java.util.stream.AbstractPipeline.copyInto(AbstractPipeline.java:481) java.util.stream.AbstractPipeline.wrapAndCopyInto(AbstractPipeline.java:471) java.util.stream.ReduceOps$ReduceOp.evaluateSequential(ReduceOps.java:708)
原因是使用Map.merge方法合并时,merge不允许value为null导致的
方法源码:
defaultVmerge(Kkey,Vvalue, BiFunctionremappingFunction){ Objects.requireNonNull(remappingFunction); //在这里判断了value不可为null Objects.requireNonNull(value); VoldValue=get(key); VnewValue=(oldValue==null)?value: remappingFunction.apply(oldValue,value); if(newValue==null){ remove(key); }else{ put(key,newValue); } returnnewValue; }
解决方案:
MapvideoGiftSumVoMap=videoGiftSum.stream().collect(Collectors.toMap (callRecodVo->Optional.ofNullable(callRecodVo).map (CallRecodVo::getStatdate).orElse(0),callRecodVo->Optional.ofNullable(callRecodVo).map (CallRecodVo::getVideogiftdiamond).orElse(0),(key1,key2)->key2));
使用optional判断空指针设置为null的默认值
分析:
因没有找到Map.merge方法为什么要检查ValueNull的相关资料和官方回答,所以做以下推断:
Collectors.toMap可以使用ConcurrentHashMap为最终收集结构,而ConcurrentHashMap不允许Value为Null避免产生二义性(ConcurrentHashMap的keyvalue不能为null,map可以?)和CAS的ABA问题,所以Map.merge为了兼容ConcurrentHashMap还有ConcurrentSkipListMap等多线程环境下使用的数据结构和使用CAS的实现不允许Value为Null
其他知识:key不能为null,是因为无法分辨是key没找到的原因所以为null,还是key值本身就为null。–key的二义性
以上为个人经验,希望能给大家一个参考,也希望大家多多支持毛票票。如有错误或未考虑完全的地方,望不吝赐教。