IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> Java集合中是否可以添加null值(或者key为null) -> 正文阅读

[数据结构与算法]Java集合中是否可以添加null值(或者key为null)

前言

面试、考试经常问的一个问题:xx集合是否可以添加null值?看到这个问题,总是拿不准,本篇就来总结下,常用数据集合是否可以添加null值(或keynull)。

1. HashTable

HashTable不允许null作为key或者value,如果任意一个为null的话,会抛出NullPointerException.

    public synchronized V put(K key, V value) {
        // Make sure the value is not null
        if (value == null) {
            throw new NullPointerException();
        }

        // Makes sure the key is not already in the hashtable.
        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或者valuenull,当keynull时,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或者valuenull,或者两者同时为null,当keynull的时候,其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或者valuenull,会在运行时抛出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();
    // code fragment...
}

可以看到,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

可能还有些许遗漏,欢迎大家在评论区补充。遇到这种问题,通过尝试或者查看源码就可以确定。

  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2022-03-16 22:43:30  更:2022-03-16 22:49:37 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/26 11:28:32-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码