应优先使用 Collection.isEmpty() 而非 size() == 0,因其时间复杂度为 O(1) 且避免懒加载开销;判空前必须先检查 null;Stream 判空推荐 anyMatch(x -> true);Optional 包裹集合时宜用 orElse(Collections.emptyList()) 统一处理。
Collection.isEmpty() 而不是 size() == 0
绝大多数集合(ArrayList、HashSet、LinkedList 等)的 isEmpty() 方法是 O(1) 时间复杂度,直接返回内部状态字段;而 size() 在某些懒加载或代理集合(如 Hibernate 的 PersistentSet)中可能触发初始化或查询,带来意外开销甚至 N+1 问题。
collection.isEmpty()
collection.size() == 0 或 collection.size() > 0
isEmpty(),需用 array.length == 0
null
Java 集合变量本身可能是 null,直接调用 isEmpty() 会抛 NullPointerException。尤其在方法参数、数据库查询结果、JSON 反序列化后字段中很常见。
if (list != null && !list.isEmpty()) {
//
安全遍历
}
Objects.firstNonNull(list, Collections.emptyList())
CollectionUtils.isNotEmpty(list) —— 它内部已处理 null 安全findAny().isPresent()
对 Stream 判空时,stream.findAny().isPresent() 是常见误用:它会消费流(不可重用),且对无限流或 IO 流有风险;更糟的是,它不短路——即使第一个元素存在,仍可能触发整个 pipeline 执行(取决于中间操作)。
List 后判空(适合小数据):!stream.collect(Collectors.toList()).isEmpty()
stream.anyMatch(x -> true) —— 短路、不消费、不新建集合anyMatch 返回 boolean,语义清晰,性能可控当集合包装在 Optional 中(如 Spring Data JPA 的 >
Optional 关联集合),容易写出双重判空嵌套,逻辑易错且可读性差。
optionalUser.map(User::getRoles)
.filter(roles -> roles != null && !roles.isEmpty())
.ifPresent(roles -> { ... });
Optional 转成“确定非空”的集合,例如用 optional.orElse(Collections.emptyList())
map 后再做 null 检查 —— 违背 Optional 设计本意null、懒加载、流式处理、Optional 嵌套这些边界交织时,不靠试错、不靠加日志,一次写对。