前言
面试、考试经常问的一个问题:xx集合是否可以添加null值?看到这个问题,总是拿不准,本篇就来总结下,常用数据集合是否可以添加null 值(或key 为null )。
1. HashTable
HashTable 不允许null 作为key 或者value ,如果任意一个为null 的话,会抛出NullPointerException .
public synchronized V put(K key, V value) {
if (value == null) {
throw new NullPointerException();
}
Entry<?,?> tab[] = table;
int hash = key.hashCode();
int index = (hash & 0x7FFFFFFF) % tab.length;
@SuppressWarnings("unchecked")
Entry<K,V> entry = (Entry<K,V>)tab[index];
for(; entry != null ; entry = entry.next) {
if ((entry.hash == hash) && entry.key.equals(key)) {
V old = entry.value;
entry.value = value;
return old;
}
}
addEntry(hash, key, value, index);
return null;
}
上面是HashTable 的源码,如果value 为空,会在参数检查时抛出空指针异常;如果key 为空,则在调用key.hashCode() 的时候发生空指针异常。
2. HashMap
HashMap 允许key 或者value 为null ,当key 为null 时,null 可以作为正常的key ,可以通过hashMap.get(null) 来获取值;null 也可以作为value 添加,均可以正常使用。
@Test
public void addNullTest() {
Map<String, Object> hashMap = new HashMap<>();
hashMap.put(null, null);
hashMap.put(null, "Hello");
hashMap.put("Hello", null);
System.out.println(hashMap.get(null));
System.out.println(hashMap.get("Hello"));
}
上述的代码输出结果为:
Hello
null
3. HashSet
HashSet 内部使用了HashMap 实现,所以HashSet 可以添加null 值。验证代码如下:
@Test
public void addNullTest() {
Set<String> hashSet = new HashSet<>();
hashSet.add(null);
hashSet.add(null);
System.out.println(hashSet.contains(null));
}
输出结果为:
true
4. Vector
Vector 内部维护了一个数组,是可以添加null 作为值的,验证代码如下:
@Test
public void addNullTest() {
Vector<String> vector = new Vector<>();
vector.add(null);
vector.add(null);
vector.add(null);
System.out.println(vector.contains(null));
}
输出结果为:
3
true
5. ArrayList
ArrayList 可以添加null 为值,验证代码如下:
@Test
public void addNullTest() {
List<String> arrayList = new ArrayList<>();
arrayList.add(null);
arrayList.add(null);
arrayList.add(null);
System.out.println(arrayList.size());
System.out.println(arrayList.contains(null));
}
输出结果为;
3
true
6. LinkedList
LinkedList 允许添加null 作为值,验证代码和ArrayList 形式一致,不贴验证代码了。
7. LinkedHashMap
LinkedHashMap 允许key 或者value 为null ,或者两者同时为null ,当key 为null 的时候,其hashCode 为0,也可以正常当作key值。下面是验证代码:
@Test
public void addNullTest() {
LinkedHashMap<String, Object> linkedHashMap = new LinkedHashMap<>();
linkedHashMap.put(null, null);
linkedHashMap.put(null, "Hello");
linkedHashMap.put("Hello", null);
System.out.println(linkedHashMap.get(null));
System.out.println(linkedHashMap.get("Hello"));
}
输出结果为:
Hello
null
8. ConcurrentHashMap
ConcurrentHashMap 不允许key 或者value 为null ,会在运行时抛出NullPointerException 。以下截取的ConcurrentHashMap 的部分源码:
public V put(K key, V value) {
return putVal(key, value, false);
}
final V putVal(K key, V value, boolean onlyIfAbsent) {
if (key == null || value == null) throw new NullPointerException();
}
可以看到,putVal 里面的参数检查,如果任意一项为空,则会抛出空指针异常。
9. ConcurrentSkipListSet
ConcurrentSkipListSet 不允许添加null 作为值,原因和ConcurrentHashMap 类似,添加元素时会对参数进行检查,如果为null会抛出空指针异常。
10. ConcurrentLinkedDeque和ConcurrentLinkedQueue
两者都不允许添加null值,因为在添加值时都会检查添加进来的值是否为null ,如果为null 会抛出NullPointerException .
11. CopyOnWriteArrayList和CopyOnWriteArraySet
这两种数据结构均允许null 值的添加。验证代码:
@Test
public void addNullTest() {
CopyOnWriteArrayList<String> copyOnWriteArrayList = new CopyOnWriteArrayList<>();
copyOnWriteArrayList.add(null);
copyOnWriteArrayList.add(null);
System.out.println(copyOnWriteArrayList.size());
System.out.println(copyOnWriteArrayList.contains(null));
CopyOnWriteArraySet<String> copyOnWriteArraySet = new CopyOnWriteArraySet<>();
copyOnWriteArraySet.add(null);
copyOnWriteArraySet.add(null);
System.out.println(copyOnWriteArraySet.size());
System.out.println(copyOnWriteArraySet.contains(null));
}
输出结果为:
2
true
1
true
12. LinkedBlockingDeque、LinkedBlockingQueue、ArrayBlockingQueue、PriorityBlockingQueue
这几个都不支持null 作为值,入参有参数检查,null 会抛出NullPointerException 。
总结
支持添加null 的集合有:
- HashMap
- HashSet
- Vector
- ArrayList
- LinkedList
- LinkedHashMap
- CopyOnWriteArrayList
- CopyOnWriteArraySet
不支持添加null 的集合有:
- HashTable
- ConcurrentHashMap
- ConcurrentSkipListSet
- ConcurrentLinkedDeque
- ConcurrentLinkedQeque
- LinkedBlockingDeque
- LinkedBlockingQueue
- ArrayBlockingQueue
- PriorityBlockingQueue
可能还有些许遗漏,欢迎大家在评论区补充。遇到这种问题,通过尝试或者查看源码就可以确定。
|