Optional 是 Java 8 引入的容器类,用于明确表达“可能为空”的值,核心用途是将空值检查从隐式运行时错误转为显式编译期契约;正确用法包括:作为方法返回类型、用 empty/of/ofNullable 创建、链式调用优先 map/flatMap、默认值优先 orElseGet 而非 orElse、禁用 isPresent()+get() 反模式。

Optional 是 Java 8 引入的容器类,用来**明确表达“可能为空”的值**,它本身不是为替代 null 而生,而是帮你把“空值检查”从隐式、易忽略的运行时错误,变成显式、强制处理的编译期契约。
用 Optional 包装可能为空的返回值
典型场景是方法可能查不到结果(如数据库查询、Map.get、Stream.findFirst)。不要直接返回 null,改用 Optional:
- ✅ 好写法:public Optional
findUserById(Long id) { return users.stream().filter(u -> u.getId().equals(id)).findFirst(); } - ❌ 避免:public User findUserById(Long id) { ... return null; }(调用方极易忘判空)
这样调用方必须主动考虑“有还是没有”,IDE 和编译器也会提醒你处理分支。
避免 new Optional() 和 Optional.of(null)
Optional 不能容纳 null 值。以下操作会抛出 NullPointerException:
立即学习“Java免费学习笔记(深入)”;
- Optional.of(null) —— 明确拒绝 null
-
new Optional
() —— 构造函数私有,无法直接实例化 - ✅ 正确创建方式:Optional.empty()(空值)、Optional.of(value)(非 null 值)、Optional.ofNullable(value)(安全,value 为 null 时自动转为 empty)
链式调用中用 map/flatMap 处理嵌套逻辑
当需要对 Optional 中的值做转换,或继续调用可能返回 Optional 的方法时,优先用 map/flatMap,而不是先 get 再判空:
- ✅ 推荐:userOpt.map(User::getProfile).map(Profile::getAvatarUrl).orElse("default.png")
- ❌ 不推荐:userOpt.get().getProfile().getAvatarUrl()(一旦中间任一环节为 null,立刻 NPE)
- flatMap 用于“Optional 返回 Optional”的场景,比如:userOpt.flatMap(u -> u.getAddressOpt())
慎用 isPresent() + get(),优先用 orElse / orElseGet / ifPresent
ifPresent() 适合“有值才执行副作用”;orElse 系列更适合提供默认值。避免写出“先判断再取值”的冗余代码:
- ✅ 清晰安全:nameOpt.orElse("anonymous")、configOpt.ifPresent(this::applyConfig)
- ❌ 反模式:if (opt.isPresent()) { use(opt.get()); }(多一次判断,且 get() 仍可能被误用)
- 注意 orElse 与 orElseGet 区别:前者无论是否为空都会执行参数表达式,后者只在为空时才执行,适合开销大的默认值构造
Optional 不是银弹,不适用于字段、参数、集合元素等场景;它最适合作为**方法返回类型**,让空值语义可读、可追踪、不可绕过。










