应主动用Objects.requireNonNull()校验入口参数,用Optional显式表达可能为空的返回值,配合@Nullable/@NonNull注解和IDE静态检查,辅以StringUtils.isBlank()等工具方法处理字符串集合判空,并通过单元测试覆盖“半空”边界场景。
Objects.requireNonNull() 主动拦截 null空指针异常(NullPointerException)多数发生在方法内部对参数或字段未校验就直接调用方法或访问属性。与其等运行时崩溃,不如在入口处明确拒绝 null。Java 7 引入的 Objects.requireNonNull() 是最轻量、语义最清晰的防御手段。
null 时立即抛出带消息的 NullPointerException,堆栈指向调用点而非深层访问点,排查更快public UserService(UserRepository repo) {
this.repo = Objects.requireNonNull(repo, "repo must not be null");
}null 场景(如可选查询结果)Optional 显式表达“可能为空”的契约当一个值天然可能不存在(比如数据库查询未命中、配置项未设置),用 null 表示会把“空”的语义隐含在调用方,极易漏判。改用 Optional 能强制调用方处理两种分支。
Optional:DAO 方法返回 Optional 而非 User,调用方必须显式调用 .isPresent()、.orElse() 或 .ifPresent()
Optional 用作字段或方法参数——它不是容器类,设计初衷仅用于返回值Optional.get():它不加判断直接取值,和直接解引用 null 风险等同;应优先用 .orElse(null) 或 .orElseThrow()
静态检查比运行时异常更早发现问题。IntelliJ 和 Eclipse 都支持基于注解推断空值流,配合编译期检查能拦截大量隐患。
@Nullable(JetBrains)或 @NonNull(AndroidX / Checker Framework),IDE 会在不安全解引用时报黄线甚至错误-Xlint:unchecked 和 -Xlint:cast 等基础检查;更进一步可集成 error-prone 或 NullAway 做编译期空值分析@RequiredArgsConstructor(onConstructor_ = {@NonNull}),可自动生成非空校验代码字符串判空、集合判空是高频操作,手写 str != null && !str.isEmpty() 容易遗漏或顺序写反,且无法复用。标准库和常用工具类已提供可靠方案。
StringUtils.isBlank(str)(Apache Commons)或 String.valueOf(obj).trim().isEmpty()(避免 NPE),禁用 str.isEmpty() 前不判 null
集合:用 CollectionUtils.isEmpty(list) 或 Java 8+ 的 list == null || list.isEmpty();避免 list.size() == 0——它在 null 时直接炸ArrayUtils.isEmpty(arr);原始类型数组(如 int[])更要小心,arr.length 对 null 同样抛 NPEnull,而是被多层封装后悄然透出的“半空”状态——比如 Optional.empty() 被误转成 null 返回,或 JSON 反序列化时字段缺失却未设默认值。这类问题不会被静态检查捕获,得靠单元测试覆盖边界输入,以及日志中记录关键对象的非空断言。