用 isEmpty() 判空最安全,但需先判 null;Map 同理;避免 size()==0 和 keySet().isEmpty() 等冗余写法;外部来源集合要区分 null 与空集合语义。

用 isEmpty() 判断集合是否为空最安全
Java 集合类(Collection 及其子接口如 List、Set、Queue)都继承了 isEmpty() 方法,它明确语义是“当前集合中是否不含任何元素”,返回 boolean。这是唯一推荐的判空方式。
常见误区是用 size() == 0,虽然多数实现下结果一致,但存在隐患:
-
size()在某些懒加载或代理集合(如 Hibernate 的PersistentSet)中可能触发额外查询或初始化,而isEmpty()通常可短路优化 - 部分不可变集合(如
ImmutableList)对isEmpty()做了常量时间优化,size()却未必 - 语义更清晰:你关心的是“有没有元素”,不是“数量是不是零”
别在 null 集合上调用 isEmpty()
如果集合变量本身为 null,直接调用 isEmpty() 会抛出 NullPointerException。必须先判 null,再判空。
推荐写法(JDK 8+):
立即学习“Java免费学习笔记(深入)”;
if (list != null && !list.isEmpty()) {
// 安全使用 list
}
或者用 Objects.nonNull() + isEmpty() 组合:
if (Objects.nonNull(list) && !list.isEmpty()) { ... }
注意:CollectionUtils.isEmpty()(Apache Commons Collections)和 CollectionUtils.isEmpty()(Spring Framework)都能同时处理 null 和空集合,但引入外部依赖需权衡。
Map 要用 isEmpty(),不是 keySet().isEmpty()
Map 不是 Collection 的子接口,但它也提供了 isEmpty() 方法,语义相同且高效。不要绕路写 map.keySet() != null && !map.keySet().isEmpty() —— 这既冗余又低效(创建中间 Set 视图)。
正确写法:
if (map != null && !map.isEmpty()) {
// 安全使用 map
}
补充说明:
-
map.size() == 0在大多数Map实现中性能与isEmpty()相当,但语义仍不如后者直接 -
map.entrySet().isEmpty()或map.values().isEmpty()同样不必要,且可能触发视图初始化
流式处理前记得先判空,避免空指针或无意义操作
用 stream() 处理集合时,若集合为 null,调用 stream() 会直接 NPE;若集合为空,stream() 返回空流,逻辑上安全,但可能掩盖上游数据缺失问题。
建议按实际语义决定:
- 仅需跳过空/空集合:先
if (list != null && !list.isEmpty()) { list.stream().... } - 需要统一处理(含
null):用Optional.ofNullable(list).filter(Collection::isEmpty).orElse(Collections.emptyList()).stream() - 用
CollectionUtils.emptyIfNull(list).stream()(Apache Commons)更简洁
最容易被忽略的是:集合变量来自外部 API、数据库查询或配置解析时,null 比空集合更常见,且两者含义不同——空集合代表“查到了但没数据”,null 往往代表“根本没查或失败”。别混为一谈。










