空指针异常是运行时逻辑错误,源于访问null引用;需从源头防控、善用Optional、初始化默认值、覆盖空路径测试,并将null视为合法状态设计。
空指针异常(NullPointerException)不是语法错误,而是运行时因访问 null 引用的成员 触发的逻辑错误。它不报在写代码时,而藏在数据未就绪、校验被跳过、或默认值未设好的缝隙里。
方法参数、返回值、集合元素、外部输入(如 JSON 解析结果、数据库查不到的记录)、以及懒加载对象,都是高危 null 来源。不要假设“它肯定不为空”——Java 不强制你检查,但 JVM 会毫不留情地抛异常。
建议:
map.get(key)、list.get(0))一律先判空,尤其 Optional.ofNullable() 可让判空更语义化Objects.requireNonNull(name, "name must not be null")
Optional 不是万能解药,但它强迫你面对“空”的可能性。它不是用来包装所有变量的,而是专用于方法返回值场景,比如查找、转换、链式计算。
正确用法示例:
Optional findName() { return Optional.ofNullable(db.loadName()); }
user.flatMap(User::getProfile).map(Profile::getEmail).orElse("no-email@example.com")
Optional.get() —— 它和直接调用一样危险;优先用 ifPresent()、orElse()、orElseGet()
实例字段、静态字段、局部变量若未显式初始化,在某些路径下就会是 null。尤其注意条件分支中遗漏赋值、try-catch 吞掉异常导致初始化失败等情况。
建议:
private List tags = new ArrayList(); ,而非 private List tags;
log.info("user: {}", user.getName()) —— 若 user 为 null,toString() 调用前就已崩溃;改用 log.info("user: {}", Objects.toString(user, "null"))
很多 NPE 出现在边界场景:数据库查无结果、HTTP 接口返回 404、前端传参缺失。这些往往不会在 happy path 测试中浮现。
建议:
when(service.findUser(123)).thenReturn(null)
在编码阶段标出潜在空引用不复杂但容易忽略:NPE 的根因从来不是“Java 不够智能”,而是我们把“假设不为空”当成了默认前提。把 null 当作一种合法状态去设计、校验和测试,异常自然就少了。