orElseGet 实现延迟加载,默认值构造成本高时更高效:orElse 总是立即执行默认值创建,而 orElseGet 仅在 Optional 为空时调用 Supplier,避免不必要的对象生成,推荐用于复杂计算、外部资源依赖等场景。

在Java中,Optional.orElseGet 是一个非常实用的方法,它能帮助我们实现延迟加载默认值,避免不必要的对象创建。相比 orElse,orElseGet 更加高效,尤其是在默认值的构造成本较高时。
orElse 与 orElseGet 的关键区别
理解两者差异是掌握懒加载技巧的基础:
- orElse(T other):无论 Optional 是否包含值,都会立即计算传入的默认值。这意味着即使有值,也会执行默认值的构造逻辑。
-
orElseGet(Supplier
supplier) :只有当 Optional 为 empty 时,才会调用 Supplier 获取默认值,实现了惰性求值。
假设我们要从数据库查询用户,查不到则创建一个默认用户对象:
// 使用 orElse —— 不推荐(可能浪费资源)
User defaultUser = new User("default", "unknown@example.com");
return Optional.ofNullable(userRepository.findById(id))
.orElse(defaultUser);
这里 new User(...) 总是会被执行,哪怕数据库查到了用户。
立即学习“Java免费学习笔记(深入)”;
// 使用 orElseGet —— 推荐(懒加载)
return Optional.ofNullable(userRepository.findById(id))
.orElseGet(() -> new User("default", "unknown@example.com"));
只有当 findById 返回 null 时,才会执行 lambda 创建默认用户,节省了对象创建开销。
何时使用 orElseGet 实现懒加载
以下场景特别适合使用 orElseGet 来延迟生成默认值:
- 默认值需要通过复杂计算获得,比如解析配置、拼接字符串。
- 默认值依赖外部资源,如读取文件、调用远程接口。
- 构造函数或工厂方法调用代价高,比如创建集合、缓存实例。
例如,获取配置项,缺失时解析默认JSON:
String jsonConfig = config.getProperty("settings");
return Optional.ofNullable(jsonConfig)
.orElseGet(() -> loadDefaultSettingsFromResource());
这里 loadDefaultSettingsFromResource() 只在必要时才调用。
常见误区与最佳实践
开发者常犯的错误是误用 orElse 导致性能问题:
- 不要在 orElse 中调用方法或新建对象,除非确定代价极小。
- Supplier 必须是无副作用的,避免 orElseGet 触发意外行为。
- 如果默认值是常量或已存在的实例,orElse 也可以接受。
正确选择取决于上下文。若默认值是静态常量:
public static final String DEFAULT_NAME = "Anonymous";
Optional.ofNullable(name).orElse(DEFAULT_NAME); // 安全且高效
基本上就这些。合理使用 orElseGet 能有效提升代码效率,特别是在处理 Optional 的 fallback 逻辑时,记得让默认值的生成“按需执行”。










