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

Optional的stream方法,flatMap, filter应用

Java 8引入的OptionalStream彻底改变了我们处理空值和集合操作的方式。本文将深入探讨如何将二者结合使用,通过四个核心场景提升代码的健壮性和简洁性。

一、Optional构成的Stream:空值自动过滤

当处理Optional集合时,我们常需要过滤掉空值并提取有效元素:

List<Optional<String>> options = Arrays.asList(Optional.of("Java"),Optional.empty(),Optional.of("Python")
);// 传统方式
List<String> values = options.stream().filter(Optional::isPresent).map(Optional::get).collect(Collectors.toList());// 更优雅方式
List<String> improved = options.stream().flatMap(Optional::stream)  // Java 9+.collect(Collectors.toList());

flatMap(Optional::stream)会自动展开非空Optional,等价于:

.flatMap(opt -> opt.map(Stream::of).orElseGet(Stream::empty))

二、安全解引用:默认值策略

避免直接使用get(),优先考虑安全解引用方式:

Optional<User> userOpt = findUserById(123);// 不安全方式(可能抛出NoSuchElementException)
User risky = userOpt.get();// 安全方式1:提供默认值
User safeUser = userOpt.orElse(new User("Guest"));// 安全方式2:延迟计算默认值
User efficientUser = userOpt.orElseGet(() -> createGuestUser());or与orElseGet方法很像,or不会解包Optional对象的值,,这一方法 9引入,不会执行其它操作,直接返回Optional对象;如果原始Optional对象为空,该方法会延迟返回一个不同的Optional对象。orElseThrow和get方法相似, Optional对象为空抛出一个异常,orElseThrow可以定制希望抛出的异常。// 安全方式3:条件消费
userOpt.ifPresentOrElse(user -> processUser(user),() -> log.warn("User not found")
);

三、Optional组合运算:安全的值组合

当需要组合多个Optional值时,可采用函数式组合:

Optional<Integer> width = Optional.of(1920);
Optional<Integer> height = Optional.of(1080);// 方式1:嵌套map(Java 8)
Optional<Resolution> res = width.flatMap(w -> height.map(h -> new Resolution(w, h))
);// 方式2:使用Tuple包装(更灵活)
record Tuple2<A,B>(A a, B b) {}
Optional<Tuple2<Integer, Integer>> dimensions = width.flatMap(w ->height.map(h -> new Tuple2<>(w, h))
);// 方式3:自定义zip方法
public static <A,B> Optional<Pair<A,B>> zip(Optional<A> a, Optional<B> b) {return a.flatMap(aVal -> b.map(bVal -> Pair.of(aVal, bVal)));
}

四、精准过滤:Optional.filter应用

在值提取前进行条件过滤:

Optional<String> emailOpt = getEmailFromRequest();// 基础过滤:非空且包含@
Optional<String> validEmail = emailOpt.filter(e -> e.contains("@"));// 组合条件过滤
Optional<Employee> manager = findEmployee(id).filter(emp -> emp.getRole().equals(Role.MANAGER)).filter(emp -> emp.getProjects().size() > 3);// 链式操作示例
String result = Optional.ofNullable(rawInput).filter(s -> !s.isBlank()).map(String::trim).filter(s -> s.length() >= 6).orElse("default");

最佳实践总结

  1. 防御性编程:始终假设Optional可能为空
  2. 早过滤原则:优先在Stream管道起始处处理空值
  3. 避免嵌套地狱:通过flatMap保持代码扁平化
  4. 语义明确:使用orElse/orElseGet明确表达空值处理策略
  5. 不变性原则:Optional应始终视为不可变容器

通过合理运用这些模式,可以使代码:

  • 减少90%以上的NullPointerException
  • 提高业务逻辑的可读性
  • 增强数据流处理的健壮性
  • 降低维护复杂度

相关文章:

Optional的stream方法,flatMap, filter应用

Java 8引入的Optional和Stream彻底改变了我们处理空值和集合操作的方式。本文将深入探讨如何将二者结合使用&#xff0c;通过四个核心场景提升代码的健壮性和简洁性。 一、Optional构成的Stream&#xff1a;空值自动过滤 当处理Optional集合时&#xff0c;我们常需要过滤掉空…...

Intellij ider部署python项目教程

自己写了一个python项目【mac电脑】&#xff0c;然后用Intellij ider打开&#xff0c;配置python解释器&#xff0c;然后一运行&#xff0c;一直报错&#xff0c; If this fails your Python may not be configured for Tk ModuleNotFoundError: No module named _tkinter 各…...

Linux进程状态补充(10)

文章目录 前言一、阻塞二、挂起三、运行R四、休眠D五、四个重要概念总结 前言 上篇内容大家看的云里雾里&#xff0c;这实在是正常不过&#xff0c;因为例如 写实拷贝 等一些概念的深层原理我还没有讲解&#xff0c;大家不用紧张&#xff0c;我们继续往下学习就行&#xff01;&…...

使用ANTLR4解析Yaml,JSON和Latex

文章目录 ANTLR4基本使用**1. 安装 Java 运行时&#xff08;必需&#xff09;****2. 安装 ANTLR4 命令行工具****方法一&#xff1a;通过包管理器&#xff08;推荐&#xff09;****macOS/Linux (Homebrew)****Windows (Chocolatey)** **方法二&#xff1a;手动安装&#xff08;…...

基于Python深度学习的鲨鱼识别分类系统

摘要&#xff1a;鲨鱼是海洋环境健康的指标&#xff0c;但受到过度捕捞和数据缺乏的挑战。传统的观察方法成本高昂且难以收集数据&#xff0c;特别是对于具有较大活动范围的物种。论文讨论了如何利用基于媒体的远程监测方法&#xff0c;结合机器学习和自动化技术&#xff0c;来…...

Ruby 简介

Ruby 简介 引言 Ruby 是一种广泛使用的动态、开源的编程语言,自 1995 年由日本程序员 Yukihiro Matsumoto(通称 Matz)设计以来,它以其优雅的语法、强大的库支持和跨平台特性赢得了全球开发者的青睐。本文将详细介绍 Ruby 的起源、特点、应用领域以及它在现代软件开发中的…...

EtherNet/IP转ProfiNet协议转换网关驱动西门子PLC与流量计的毫秒级压力同步控制

一、案例背景 汽车涂装线的静电喷涂工艺对压缩空气流量稳定性要求极高。原系统中Alicat流量计与西门子PLC因协议差异无法联动&#xff0c;导致涂料浪费率高达8%。通过JM-EIPM-PN网关实现供气系统与PLC的深度集成。从而实现了EtherNet/IP转ProfiNet的通讯。 二、设备连接与配置…...

为mariadb和mysql添加用户和修改密码的方法

一、查看MariaDB中的用户 步骤1&#xff1a;登录MariaDB sudo mysql -u root -p # 使用root账户登录&#xff08;输入密码&#xff09; 步骤2&#xff1a;查询用户列表 -- 切换到mysql系统数据库 USE mysql; -- 查看所有用户及其主机权限 SELECT User, Host FROM user; 输出…...

【力扣刷题|第十七天】0-1 背包 完全背包

目标和 力扣题目网址:目标和 这道题我们先用回溯的思想来做。首先我们设正数和为S&#xff0c;数组和为N&#xff0c;目标值为T&#xff0c;那么S-(N-S)T化简之后可以得S(TN)/2即选择的正数个数为偶数&#xff0c;而且NT也为偶数&#xff0c;那么第一个判断条件我们就有了&…...

python的内存管理

目录 1. 引用计数 2. 垃圾收集&#xff08;GC&#xff09; python的内存管理主要是引用计数和垃圾回收器来进行内存管理 1. 引用计数 每个 Python 对象都有一个引用计数&#xff0c;当引用计数为零时&#xff0c;对象的内存会被释放。 import sysa [] # 创建一个空列表对…...

TDengine 中的异常恢复

简介 本章主要介绍在 TDengine 执行命令过程中发生异常&#xff0c;如何手工终于执行的任务。可以终止连接&#xff0c;线上查询及终止事务。 如果一个事务 在一个复杂的应用场景中&#xff0c;连接和查询任务等有可能进入一种错误状态或者耗时过长迟迟无法结束&#xff0c;…...

SQL Server:当在删除数据库时因为存在触发器而无法删除

当在删除数据库时因为存在触发器而无法删除&#xff0c;你可以通过禁用触发器来解决这个问题。下面为你介绍在 SQL Server 里禁用和启用触发器的方法。 禁用数据库中所有表的触发器 你可以使用系统视图 sys.triggers 来查询数据库里所有的触发器&#xff0c;然后生成禁用这些…...

深度学习处理时间序列(3)

基于常识、不使用机器学习的基准 在开始使用像黑盒子一样的深度学习模型解决温度预测问题之前&#xff0c;我们先尝试一种基于常识的简单方法。它可以作为一种合理性检查&#xff0c;还可以建立一个基准&#xff0c;更高级的机器学习模型需要超越这个基准才能证明其有效性。对…...

C++11 -表达式/包装器

1. lambda 表达式各部分说明 [capture-list]&#xff1a;捕捉列表&#xff0c;该列表总是出现在 lambda 函数的开始位置&#xff0c;编译器根据 [] 来判断接下来的代码是否为 lambda 函数&#xff0c;捕捉列表能够捕捉上下文中的变量供 lambda 函数使用。(parameters)&#xf…...

VectorBT:使用PyTorch+LSTM训练和回测股票模型 进阶二

VectorBT&#xff1a;使用PyTorchLSTM训练和回测股票模型 进阶二 本方案基于LSTM神经网络构建多时间尺度股票收益率预测模型&#xff0c;结合VectorBT进行策略回测。核心原理是通过不同时间窗口&#xff08;5/10/20/30日&#xff09;捕捉股价的短期、中期、长期模式&#xff0c…...

蓝桥杯 第十二天 819 递增序列

注意注意&#xff1a;不考虑左上的情况&#xff0c;因为题目给的样例没有 public static int is1(char ch[][],int m,int n){int ans0;for (int i0;i<m;i){//起始点在哪for (int j0;j<n;j){int add1;while(jadd<n){if(ch[i][j]<ch[i][jadd]) ans; //横add;}add1…...

ThreadLocal 的妙用(线程隔离)与陷阱(内存泄漏)

前言 在Java开发中&#xff0c;线程安全是一个高频关键词。当我们使用多线程处理共享数据时&#xff0c;常常需要加锁或使用同步机制来避免数据混乱。但有一把“锁”却能让每个线程拥有自己的独立数据副本&#xff0c;它就是ThreadLocal。接下来通过实际案例&#xff0c;带你理…...

【YOLOv11】目标检测任务-实操过程

目录 一、torch环境安装1.1 创建虚拟环境1.2 启动虚拟环境1.3 安装pytorch1.4 验证cuda是否可用 二、yolo模型推理2.1 下载yolo模型2.2 创建模型推理文件2.3 推理结果保存路径 三、labelimg数据标注3.1 安装labelimg3.2 解决浮点数报错3.3 labelimg UI界面介绍3.4 数据标注案例…...

C++_STL之vector篇

一、vector的常见用法 注&#xff1a;C中若使用vector需包含头文件<vector>. 1.vector的构造函数 int n 10,ret1;vector<int> nums(n,ret); //n表示vector初始的容量 ret表示vector中初始化的值for (auto e : nums)cout << e << " "; 扩展…...

Imgui处理glfw的鼠标键盘的方法

在Imgui初始化时&#xff0c;会重新接手glfw的键盘鼠标事件。也就是遇到glfw的键盘鼠标事件时&#xff0c;imgui先会运行自己的处理过程&#xff0c;然后再去处理用户自己注册的glfw的键盘鼠标事件。 看imgui_impl_glfw.cpp源码的安装回调函数部分代码 void ImGui_ImplGlfw_In…...

【松子悲剧的七层本质】

松子悲剧的七层本质 核心命题&#xff1a;松子的悲剧是人性与社会结构的双重绞杀&#xff0c;本质是“爱的异化”与“自我消解”的终极悖论。 第一层&#xff1a;家庭缺爱的童年烙印 现象&#xff1a;松子因父亲对病弱妹妹的偏爱&#xff0c;长期处于情感匮乏状态&#xff0c;…...

sqli-labs靶场 less 9

文章目录 sqli-labs靶场less 9 时间盲注 sqli-labs靶场 每道题都从以下模板讲解&#xff0c;并且每个步骤都有图片&#xff0c;清晰明了&#xff0c;便于复盘。 sql注入的基本步骤 注入点注入类型 字符型&#xff1a;判断闭合方式 &#xff08;‘、"、’、“”&#xf…...

2-1 MATLAB鮣鱼优化算法ROA优化LSTM超参数回归预测

本博客来源于CSDN机器鱼&#xff0c;未同意任何人转载。 更多内容&#xff0c;欢迎点击本专栏目录&#xff0c;查看更多内容。 目录 0.ROA原理 1.LSTM程序 2.ROA优化LSTM 3.主程序 4.结语 0.ROA原理 具体原理看原文&#xff0c;但是今天咱不用知道具体原理&#xff0c;只…...

fircrawl本地部署

企业内部的网站作为知识库给dify使用&#xff0c;使用fircrawl来爬虫并且转换为markdown。 ​ git clone https://github.com/mendableai/firecrawl.gitcd ./firecrawl/apps/api/ cp .env.example .env cd ~/firecrawl docker compose up -d 官方&#xff1a; https://githu…...

Labview学习记录

1.快捷键 ctrlR 运行 ctrlB 去除断线 ctrlH 即时帮助 ctrlE 前后面板切换 2.画面移动 ctrlshift鼠标左键...

【Golang】第八弹----面向对象编程

&#x1f525; 个人主页&#xff1a;星云爱编程 &#x1f525; 所属专栏&#xff1a;Golang &#x1f337;追光的人&#xff0c;终会万丈光芒 &#x1f389;欢迎大家点赞&#x1f44d;评论&#x1f4dd;收藏⭐文章 前言&#xff1a;Go语言面向对象编程说明 Golang也支持面向对…...

java基础以及内存图

java基础 命名&#xff1a; 大驼峰&#xff1a;类名 小驼峰&#xff1a;变量名方法名等其他的 全部大写&#xff1a;常量名字.. // 单行注释 /**/ 多行注释 变量类型 变量名 一、基本类型&#xff08;8个&#xff09; 整数&#xff1a;byte-8bit short-16bit int 32-b…...

【嵌入式学习3】TCP服务器客户端 - UDP发送端接收端

目录 1、TCP TCP特点 TCP三次握手&#xff08;建立TCP连接&#xff09;&#xff1a; TCP四次握手【TCP断开链接的时候需要经过4次确认】&#xff1a; TCP网络程序开发流程 客户端开发&#xff1a;用户设备上的程序 服务器开发&#xff1a;服务器设备上的程序 2、UDP 为…...

Linux之基础知识

目录 一、环境准备 1.1、常规登录 1.2、免密登录 二、Linux基本指令 2.1、ls命令 2.2、pwd命令 2.3、cd命令 2.4、touch命令 2.5、mkdir命令 2.6、rmdir和rm命令 2.7man命令 2.8、cp命令 2.9、mv命令 2.10、cat命令 2.11、echo命令 2.11.1、Ctrl r 快捷键 2…...

llamafactory微调效果与vllm部署效果不一致如何解决

在llamafactory框架训练好模型之后&#xff0c;自测chat时模型效果不错&#xff0c;但是部署到vllm模型上效果却很差 这实际上是因为llamafactory微调时与vllm部署时的对话模板不一致导致的。 对应的llamafactory的代码为 而vllm启动时会采用大模型自己本身设置的对话模板信息…...