Java集合对null支持差异显著:ArrayList/LinkedList允许任意位置存null;HashMap/LinkedHashMap允许一个null键和多个null值;TreeMap/TreeSet完全禁止null;ConcurrentHashMap拒绝null键值,CopyOnWriteArrayList允许add(null)。
Java集合对 null 的支持差异很大,不是所有集合都允许存 null,也不是所有位置都允许——关键看具体实现类及其设计意图。
这两个基于数组或链表的动态列表,对 null 完全开放。你可以往任意索引插入 null,也能用 null 作为元素调用 contains() 或 indexOf()。
list.add(null) 合法,长度+1list.set(0, null) 合法,替换原元素list.indexOf(null) 返回第一个 null 的下标(若存在)Arrays.asList(null) 创建的“伪List”不支持 add(),和底层是数组有关哈希表实现允许一个 null 键(因 null.hashCode() 被特殊处理),也允许多个 null 值。
map.put(null, "a") 合法,且后续再 put(null, "b") 会覆盖map.put("k", null) 合法,value 为 null 不影响查找map.get(null) 正常返回对应 value;map.containsKey(null) 也正常工作红黑树结构依赖元素的自然顺序或比较器,而 null.compareTo(x) 或 comparator.compare(null, x) 必抛 NullPointerEx。
ception
treeMap.put(null, "v") → 直接抛 NPEtreeSet.add(null) → 同样抛 NPEnull,仍可能出错Comparator.nullsFirst() / nullsLast()(Java 8+)出于线程安全和内部逻辑健壮性考虑,这些并发集合主动拒绝 null。
concurrentMap.put(null, "v") → 抛 NullPointerException
concurrentMap.put("k", null) → 同样抛 NPE(value 也不行)copyOnWriteList.add(null) 合法(它继承自 AbstractList,未重写 null 检查)copyOnWriteList.set(0, null) 是允许的,因为它不涉及结构性修改检查实际编码中别靠记忆硬背,遇到不确定的集合类型,直接查其 JDK 文档的 “Throws” 小节——几乎所有集合实现都会明确写出是否抛 NPE 及触发条件。不复杂但容易忽略。