ConcurrentHashMap为什么不允许插入null值?

在Java语言中,给ConcurrentHashMap和HashTable这些线程安全的集合中的key或者value插入null(空)值时,会报空指针异常,但是单线程操作的HashMap又运行key或者 value插入null(空)

1、查看源码

源码如下:

ConcurrentHashMap为什么不允许插入null值?_第1张图片

从源码来看,在key或者value为null(空)时,直接抛出空指针异常

为什么ConcurrentHashMap 的key或者value不允许插入null(空),而HashMap允许插入那?

2、问题分析

因为给ConcurrentHashMap加null(空)值会存在歧义,假设ConcurrentHashMap允许存储null(空)那么,我们取值的时候会出现两种结果:

1)值没有在集合中,取出的是null

2)值就是null,所以返回的结果就是原本的null值

这种情况下就会出现歧义

HashMap允许插入null(空值),是因为HashMap是针对单线程设计的,所以在取到null(空)时我们可以通过HashMap的containsKey(key)区分这个null(空)值是插入的null值,还是本来就没有返回的null(空);

那么就有人会有疑问了,ConcurrentHashMap里面也有containsKey();是不是可以区分那?

在单线程的情况下这个是没有问题的,但是ConcurrentHashMap是针对多线程的,多线程的情况下情况比较复杂的。

假如我们有两个线程T1和T2

当T1线程调用ConcurrentHashMap的containsKey(key)判断结果为false,说明T1并没有向ConcurrentHashMap 中put   null(空)值;然后在T1线程还没返回结果之前:T2线程调用了ConcurrentHashMap的put方法,插入一个key,并存入的value是一个null(空)值进来,那么T1线程调用ConcurrentHashMap的containsKey(key)的结果就是true了。

显然这个结果和我们之前期望的值false是不同的,

也就是说在多线程的复杂的情况下,到底是插入的null(空)值,还是本来就没有此key值返回的null(空)值,会存在歧义

3、总结

ConcurrentHashMap不允许插入null(空)值,主要是为了避免在并发场景下出现歧义的问题(源码中直接抛出异常的)。

你可能感兴趣的:(java,多线程和高并发,java,开发语言)