为啥 Java 中不推荐将 Optional 当做参数使用?

2023-11-09 23:40

本文主要是介绍为啥 Java 中不推荐将 Optional 当做参数使用?,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、背景

最近开发过程中,身边的同事为了实现逻辑复用,定义一个私有公共方法实现逻辑复用,定义函数签名时将上游的 Optional 作为参数传递。
IDEA 给出警告,但是并没有讲清楚为什么。

效果如下:
在这里插入图片描述

Optional 怎么使用不是本文的重点,如果想掌握可以参考 《 J a v a 8 实 战 》 「 1 」 《Java 8 实战》^「1」 Java81 自行学习。

本文主要聊为什么不让作为参数使用。
在这里插入图片描述

工作过几年的人能够发现一个规律,线上出现的异常很大比例都是空指针。

Java 8 引入 Optional 主要是为了避免出现空指针;避免代码中出现各种 null 检查等。

那么,为什么不推荐作为参数使用呢?

二、讨论

2.1 为什么不要将 Optional 作为参数

如果将 Optional 当做参数使用,那么本身可传递 null, 依然需要进行判空再使用。
并不能有效避免空指针,甚至带来额外的判断。


案例1:
直接使用 String :

public String doSomething(String name) {if (name == null) {return "你好";} else {return "你好 " + name;}
}

使用 Optional 作参数:

public String doSomething(Optional<String> name) { if (name == null || !name.isPresent()) {return "你好";} else {return "你好" + name;}
}

示例2:
由于我们通常都是将 Optional 当做返回值使用,潜意识认为不会传递 null, 通常就直接使用:

public static List<Person> search(List<Person> people, String name, Optional<Integer> age) {if(CollectionUtils.isEmpty(people)|| null == name ){return new ArrayList<>();}return people.stream().filter(p -> p.getName().equals(name)).filter(p -> p.getAge().get() >= age.orElse(0)).collect(Collectors.toList());
}

如果代码比较复杂,其他程序员不容易注意到这点,他可能会认为不需要校验 age ,因此就传 null:

someObject.search(people, "Peter", null);

结果造成了空指针!!

public static List<Person> search(List<Person> people, String name, Integer age) {if(CollectionUtils.isEmpty(people)|| null == name ){return new ArrayList<>();}final Integer ageFilter = age != null ? age : 0;return people.stream().filter(p -> p.getName().equals(name)).filter(p -> p.getAge().get() >= ageFilter).collect(Collectors.toList());
}

因此,尽量避免将 Optional 作为参数使用。

本质上是 Optional 作参数时,上游通常可以自己构建 Optional 或者取下游某个调用的返回值传递。

当使用某个调用返回值传递时,通常不会出现空指针,但是自己去执行调用传递 null 时很容易出现空指针。

2.2 非要当做参数怎么办?

有些场景希望直接将下游的返回值作为参数传递。

模拟示例如下:

    private static String first(String someParam){return   something( "first",  someParam, invokeSomeFunction(someParam));}private static String second(String someParam){return   something( "second",  someParam, invokeOtherFunction(someParam));}private  static <T> T something(String name ,String someParam,Optional<T> optional){// 各种公共逻辑return optional.get();}// 模拟下游接口1private static Optional<String> invokeSomeFunction(String someParam){return Optional.of(someParam);}// 模拟下游接口2private static Optional<String> invokeOtherFunction(String someParam){return Optional.of(someParam);}

下游返回 Optional<String>是合理的,但我们又不能将 Optional<String> 作为参数传递。

因此有如下写法:

    private static String first(String someParam){return   something( "first",  someParam, invokeSomeFunction(someParam).orElse(null));}private static String second(String someParam){return   something( "second",  someParam, invokeOtherFunction(someParam).orElse(null));}private  static <T> T something(String name ,String someParam,T param){// 各种公共逻辑return null;}

如果自定义方法过多,都要 orElse 去转为非 Optional 对象,显然不太优雅。

其实,这种场景本质上是希望将调用作为参数传递下去,因此想到了直接使用 Supplier 或者 Function 等。

   private static String first(String someParam){return   something( "first",  someParam, ()->invokeSomeFunction(someParam));}private static String second(String someParam){return   something( "second",  someParam, ()->invokeOtherFunction(someParam));}private  static <T> T something(String name , String someParam, Supplier<Optional<T>> optional){// 各种公共逻辑return  null;}

这样 Optional 依然是作为返回值使用,参数是方法调用 Supplier 也不违规,又契合将调用传递的目的。

2.3 Optional 不是万能的

Optional虽然能够减少空指针,但是滥用也会降低代码可读性。

Optional本身没有实现序列化接口,做属性时,如果使用 JDK 序列化将会报错。
可以使用 guava 包里的 Optional类替代。

三、结论

【建议】不建议将 Optional 作为参数,容易造成空指针和误解,这和 Optional 的目的相违背。如果是想传递某个调用,请使用 Supplier。
【建议】不建议将 Optional 作为属性,非要用建议使用 guava 包的 Optional 类。

四、拓展阅读

《深入理解 Lambda 表达式》

《Lambda 表达式带来的复杂性的破解之道》

参考文献

[1] 厄马(Raoul-Gabriel Urma) / 弗斯科(Mario Fusco) / 米克罗夫特(Alan Mycroft)《Java 8 实战》.人民邮电出版社
[2] https://rules.sonarsource.com/java/RSPEC-3553
[3] https://www.baeldung.com/java-optional

这篇关于为啥 Java 中不推荐将 Optional 当做参数使用?的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/379137

相关文章

Spring Boot 实现 IP 限流的原理、实践与利弊解析

《SpringBoot实现IP限流的原理、实践与利弊解析》在SpringBoot中实现IP限流是一种简单而有效的方式来保障系统的稳定性和可用性,本文给大家介绍SpringBoot实现IP限... 目录一、引言二、IP 限流原理2.1 令牌桶算法2.2 漏桶算法三、使用场景3.1 防止恶意攻击3.2 控制资源

Mac系统下卸载JAVA和JDK的步骤

《Mac系统下卸载JAVA和JDK的步骤》JDK是Java语言的软件开发工具包,它提供了开发和运行Java应用程序所需的工具、库和资源,:本文主要介绍Mac系统下卸载JAVA和JDK的相关资料,需... 目录1. 卸载系统自带的 Java 版本检查当前 Java 版本通过命令卸载系统 Java2. 卸载自定

springboot下载接口限速功能实现

《springboot下载接口限速功能实现》通过Redis统计并发数动态调整每个用户带宽,核心逻辑为每秒读取并发送限定数据量,防止单用户占用过多资源,确保整体下载均衡且高效,本文给大家介绍spring... 目录 一、整体目标 二、涉及的主要类/方法✅ 三、核心流程图解(简化) 四、关键代码详解1️⃣ 设置

Java Spring ApplicationEvent 代码示例解析

《JavaSpringApplicationEvent代码示例解析》本文解析了Spring事件机制,涵盖核心概念(发布-订阅/观察者模式)、代码实现(事件定义、发布、监听)及高级应用(异步处理、... 目录一、Spring 事件机制核心概念1. 事件驱动架构模型2. 核心组件二、代码示例解析1. 事件定义

SpringMVC高效获取JavaBean对象指南

《SpringMVC高效获取JavaBean对象指南》SpringMVC通过数据绑定自动将请求参数映射到JavaBean,支持表单、URL及JSON数据,需用@ModelAttribute、@Requ... 目录Spring MVC 获取 JavaBean 对象指南核心机制:数据绑定实现步骤1. 定义 Ja

python使用库爬取m3u8文件的示例

《python使用库爬取m3u8文件的示例》本文主要介绍了python使用库爬取m3u8文件的示例,可以使用requests、m3u8、ffmpeg等库,实现获取、解析、下载视频片段并合并等步骤,具有... 目录一、准备工作二、获取m3u8文件内容三、解析m3u8文件四、下载视频片段五、合并视频片段六、错误

javax.net.ssl.SSLHandshakeException:异常原因及解决方案

《javax.net.ssl.SSLHandshakeException:异常原因及解决方案》javax.net.ssl.SSLHandshakeException是一个SSL握手异常,通常在建立SS... 目录报错原因在程序中绕过服务器的安全验证注意点最后多说一句报错原因一般出现这种问题是因为目标服务器

CSS Anchor Positioning重新定义锚点定位的时代来临(最新推荐)

《CSSAnchorPositioning重新定义锚点定位的时代来临(最新推荐)》CSSAnchorPositioning是一项仍在草案中的新特性,由Chrome125开始提供原生支持需... 目录 css Anchor Positioning:重新定义「锚定定位」的时代来了! 什么是 Anchor Pos

gitlab安装及邮箱配置和常用使用方式

《gitlab安装及邮箱配置和常用使用方式》:本文主要介绍gitlab安装及邮箱配置和常用使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1.安装GitLab2.配置GitLab邮件服务3.GitLab的账号注册邮箱验证及其分组4.gitlab分支和标签的

Java实现删除文件中的指定内容

《Java实现删除文件中的指定内容》在日常开发中,经常需要对文本文件进行批量处理,其中,删除文件中指定内容是最常见的需求之一,下面我们就来看看如何使用java实现删除文件中的指定内容吧... 目录1. 项目背景详细介绍2. 项目需求详细介绍2.1 功能需求2.2 非功能需求3. 相关技术详细介绍3.1 Ja