为什么ConcurrentHashMap的keyvalue不能为null,map可
以?
源码
if (key == null || value == null) throw new NullPointerException();
⼆义性
假定ConcurrentHashMap也可以存放value为null的值。那不管是HashMap还是ConcurrentHashMap调⽤(key)的时候,如果返回了null,那么这个null,
都有两重含义:
1.这个key从来没有在map中映射过。
2.这个key的value在设置的时候,就是null。
为什么map允许value=null
对于HashMap的正确使⽤场景是在单线程下使⽤。
在单线程中,当我们得到的value是null的时候,我可以⽤ainsKey(key)⽅法来区分上⾯说的两重含义。
所以当(key)返回的值是null,在HashMap中虽然存在⼆义性,但是结合containsKey⽅法可以避免⼆义性。
为什么ConcurrentHashMap不允许nullpointerexception为什么异常
ConcurrentHashMap的使⽤场景为多线程。
⽤反证法来推理,假设concurrentHashMap允许存放值为null的value。
这时有A、B两个线程。
线程A调⽤(key)⽅法,返回为null,我们还是不知道这个null是没有映射的null还是存的值就是null。
我们假设此时返回为null的真实情况就是因为这个key没有在map⾥⾯映射过。那么我们可以⽤
但是在我们调⽤(key)⽅法之后,containsKey⽅法之前,有⼀个线程B执⾏了concurrentHashMap.put(key,null)的操作。那么我们调⽤containsKey⽅法返回的就是true了。这就与我们的假设的真实情况不符合了。也就是上⾯说的⼆义性。
对于key不能为null
源码就是这样。。
补充:Hashtable/HashMap与key/value为null的关系
1、 HashMap计算key的hash值时调⽤单独的⽅法,在该⽅法中会判断key是否为null,如果是则返回0;⽽Hashtable中则直接调⽤key的hashCode()⽅法,因此如果key为null,则抛出空指针异常。
2、 HashMap将键值对添加进数组时,不会主动判断value是否为null;⽽Hashtable则⾸先判断value是否为null。
3、以上原因主要是由于Hashtable继承⾃Dictionary,⽽HashMap继承⾃AbstractMap。
4、虽然ConcurrentHashMap也继承⾃AbstractMap,但是其也过滤掉了key或value为null的键值对。
以上为个⼈经验,希望能给⼤家⼀个参考,也希望⼤家多多⽀持。如有错误或未考虑完全的地⽅,望不吝赐教。