当前位置: 首页 > article >正文

Java 中Supplier延迟生成值的原因

在编程中,延迟生成值(Lazy Value Generation) 是指将值的计算或生成过程推迟到真正需要使用该值时才执行。这一机制的核心是避免不必要的计算,提升程序的性能和资源利用率。结合 SupplierOptional 的使用场景,我们可以从以下几个方面理解延迟生成值:

一、核心概念:为什么需要延迟?

假设你需要一个默认值,但这个默认值的生成可能:

  • 耗时(如数据库查询、网络请求)。
  • 耗资源(如创建大型对象、复杂计算)。
  • 依赖状态(如需要先判断某个条件是否成立)。

如果提前生成默认值,可能会导致:

  • 性能浪费:即使最终不需要使用默认值(比如 Optional 中有值),也会白白执行计算。
  • 逻辑错误:默认值的生成可能依赖后续才会变化的状态。

延迟生成的目标:仅在必要时执行计算,减少无意义的开销。

二、通过 Supplier 实现延迟生成

Supplier 是 Java 函数式接口(@FunctionalInterface),仅定义一个无参数方法 get(),用于返回一个值。它的关键作用是:将值的生成逻辑封装在 get(),而不立即执行。

示例:模拟耗时操作
import java.util.Optional;
import java.util.UUID;public class LazyDemo {public static void main(String[] args) {Optional<String> optional = Optional.empty(); // 假设无值// 错误写法:提前执行生成逻辑(无论是否需要)String eagerValue = generateEagerly(); // 立即执行String value1 = optional.orElse(eagerValue); // 正确写法:延迟执行生成逻辑(仅在必要时)String value2 = optional.orElseGet(() -> generateLazily()); // 仅当optional为空时执行}// 提前生成(立即执行)private static String generateEagerly() {System.out.println("执行提前生成逻辑");return UUID.randomUUID().toString();}// 延迟生成(通过Supplier封装)private static String generateLazily() {System.out.println("执行延迟生成逻辑");return UUID.randomUUID().toString();}
}
执行结果对比
  • optional 为空时:
    • generateEagerly() 会在调用 orElse 前就执行,输出 执行提前生成逻辑
    • generateLazily() 仅在调用 orElseGetoptional 为空时执行,输出 执行延迟生成逻辑
  • optional 有值时(如 Optional.of("有值")):
    • generateEagerly() 仍然会执行(浪费计算)。
    • generateLazily() 不会执行(节省资源)。

三、延迟生成的典型应用场景

1. 避免不必要的计算
// 假设getExpensiveData()是耗时操作
Optional<String> data = fetchData(); // 错误:无论data是否有值,都会执行getExpensiveData()
String result = data.orElse(getExpensiveData()); // 正确:仅当data为空时,才执行getExpensiveData()
String result = data.orElseGet(() -> getExpensiveData()); 
2. 依赖上下文的动态默认值
public class UserService {private final Database database;public User getUser(String userId) {Optional<User> user = database.queryUser(userId);// 仅当用户不存在时,创建默认用户(依赖当前时间)return user.orElseGet(() -> createDefaultUser(LocalDateTime.now()));}private User createDefaultUser(LocalDateTime timestamp) {// 需要当前时间作为参数,延迟生成时才能获取最新值return new User("default", timestamp);}
}
  • 如果提前生成默认用户(如 orElse(createDefaultUser(...))),可能使用过时的时间戳。
  • 延迟生成时,createDefaultUser 能获取当前最新的上下文数据(如实时时间)。
3. 流式编程中的延迟计算

在 Java Stream 中,某些操作(如 filtermap)是惰性的(Lazy),仅在终端操作(如 forEachcollect)时才执行。Supplier 可以配合这种特性优化性能:

List<String> data = ...;// 延迟生成随机数,仅在需要时(如过滤后的数据满足条件)才生成
data.stream().filter(s -> s.startsWith("A")).findFirst().orElseGet(() -> UUID.randomUUID().toString()); // 仅当无匹配元素时生成

四、延迟生成与立即生成的对比

场景延迟生成(orElseGet立即生成(orElse
性能仅在必要时执行计算,节省资源无论是否需要,均提前执行计算
适用场景默认值生成耗时、依赖动态状态默认值已提前存在或生成成本极低
代码可读性需通过 lambda 封装逻辑,稍显复杂直接传入值,代码更简洁
线程安全性每次调用 get() 都会重新计算(需注意)提前生成一次,后续使用同一值

五、总结:延迟生成的本质

延迟生成值的核心是将计算逻辑与值的使用时机解耦,通过 Supplier 等工具将 “生成值” 的动作推迟到真正需要的时刻。这一思想在 Java 中广泛应用于:

  • Optional 的默认值生成(orElseGet)。
  • 流式编程的惰性操作。
  • 框架中的依赖注入(如 Spring 的 Lazy 注解)。
  • 缓存场景的按需加载(如 ConcurrentHashMap.computeIfAbsent)。

通过延迟生成,可以显著提升程序的效率,避免 “过早计算” 带来的性能损耗,尤其在处理高成本操作或动态依赖时优势明显。

http://www.lryc.cn/news/2379879.html

相关文章:

  • 设置windows10同时多用户登录方法
  • Web 技术与 Nginx 网站环境部署
  • 分组背包问题:如何最大化背包价值?
  • nodejs快速入门到精通1
  • FP8精度革命:Hopper架构下大模型训练的误差传播控制方法
  • 手动制做一个Transformer
  • 已解决——如何让网站实现HTTPS访问?
  • WebRTC技术EasyRTC嵌入式音视频通信SDK助力智能电视搭建沉浸式实时音视频交互
  • Unreal Engine: Windows 下打包 AirSim项目 为 Linux 平台项目
  • Spring MVC HttpMessageConverter 的作用是什么?
  • 小乌龟git中的推送账户、作者账户信息修改
  • Kubernetes MCP服务器(K8s MCP):如何使用?
  • Node.js聊天室开发:从零到上线的完整指南
  • R²AIN SUITE 亮相第九届智能工厂高峰论坛
  • 深入理解仿函数(Functors):从概念到实践
  • InternLM 论文分类微调实践(XTuner 版)
  • 《Python星球日记》 第88天:ChatGPT 与 LangChain
  • PC:使用WinSCP密钥文件连接sftp服务器
  • 1688正式出海,1688跨境寻源通接口接入,守卫的是国内工厂资源
  • 力扣303 区域和检索 - 数组不可变
  • Spring的后置处理器是干什么用的?扩展点又是什么?
  • [ linux-系统 ] 进程地址空间
  • 文件名是 ‪E:\20250512_191204.mp4, EV软件录屏,未保存直接关机损坏, 如何修复?
  • Java常见API文档(下)
  • DRIVEGPT4: 通过大语言模型实现可解释的端到端自动驾驶
  • 知识图谱(KG)与大语言模型(LLM)
  • 构建共有语料库 - Wiki 语料库
  • 苍穹外卖项目中的 WebSocket 实战:实现来单与催单提醒功能
  • 精益数据分析(59/126):移情阶段的深度博弈——如何避开客户访谈的认知陷阱
  • Win10 安装单机版ES(elasticsearch),整合IK分词器和安装Kibana