一、背景

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

效果如下:

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

本文主要聊为什么不让作为参数使用。

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

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 当做参数使用?相关推荐

  1. [转载] Java中方法不可以有默认参数

    参考链接: Java中的方法 Java中的方法不可以有默认参数,只能通过重载来实现: public class ParameterDefault { /** * @param args */ publ ...

  2. java中play方法的两个参数_如何避免在play2中到处传递参数?

    在我看来,模板是静态类型的这一事实实际上是一件好事:你可以保证调用你的模板如果编译就不会失败. 但是,它确实在调用站点上添加了一些样板.但是你可以减少它(不会失去静态类型优势). 在Scala中,我看 ...

  3. 为什么说 Java 中只有值传递?

    对于初学者来说,要想把这个问题回答正确,是比较难的.在第二天整理答案的时候,我发现我竟然无法通过简单的语言把这个事情描述的很容易理解,遗憾的是,我也没有在网上找到哪篇文章可以把这个事情讲解的通俗易懂. ...

  4. 为什么说Java中只有值传递(另一种角度)

    转载自 为什么说Java中只有值传递 对于初学者来说,要想把这个问题回答正确,是比较难的.在第二天整理答案的时候,我发现我竟然无法通过简单的语言把这个事情描述的很容易理解,遗憾的是,我也没有在网上找到 ...

  5. 为什么Java中只有值传递

    原文链接:https://www.cnblogs.com/wchxj/p/8729503.html 在开始深入讲解之前,有必要纠正一下大家以前的那些错误看法了.如果你有以下想法,那么你有必要好好阅读本 ...

  6. 什么是值传递,什么是引用传递。为什么说Java中只有值传递。

    关于这个问题,在StackOverflow上也引发过广泛的讨论,看来很多程序员对于这个问题的理解都不尽相同,甚至很多人理解的是错误的.还有的人可能知道Java中的参数传递是值传递,但是说不出来为什么. ...

  7. 为什么说Java中只有值传递

    点击上方"程序员小灰",选择"置顶公众号" 有趣有内涵的文章第一时间送达! 本文转载自公众号 Hollis 对于初学者来说,要想把这个问题回答正确,是比较难的. ...

  8. Java中的main方法

    2019独角兽企业重金招聘Python工程师标准>>> 在一个Java应用程序中,通常程序的入口是一个main方法,它被声明为公有静态方法,参数是一个字符串数组,返回值为Void类型 ...

  9. java中的逻辑结构

    本文将介绍java中的三种逻辑结构:顺序结构.分支结构.循环结构 目录 一.顺序结构 二.分支结构 1.if语句 1.1单分支语句 1.2双分支语句 1.3多分支语句 2.switch语句 三.循环结 ...

  10. java 物理内存_聊聊Java中的内存

    JVM的内存 先放一张JVM的内存划分图,总体上可以分为堆和非堆(粗略划分,基于java8) 那么一个Java进程最大占用的物理内存为: Max Memory = eden + survivor + ...

最新文章

  1. 中国电信换将 三家运营商未来将如何争战
  2. FPGA之道(60)时空变换之空域优化
  3. Android定位开发之百度定位、高德定位、腾讯定位,三足鼎立一起为我所用!
  4. linux-centerOs6.8安装nginx与配置
  5. Scala集合:ListBuffer可变集合的head/tail/last/init方法
  6. php记录代码执行时间
  7. python的matplotlib生成colorbar
  8. cjson 对象是json数组型结构体_C语言 - cJSON解析特定格式 含有数组array类型的数据...
  9. python课后题答案第二章_Python编程:从入门到实践——练习题答案(第二章)
  10. Android_组件_BroadcastReceiver基础
  11. Markdown(四)——绘图工具mermaid之状态图stateDiagram
  12. 原创 leetcode[454]四数相加II /4Sum II 哈希策略
  13. js获取html中div里的标签id_【学废了】HTML初步
  14. 通达信波段王指标公式主图_通达信股票波段王+精准买卖提示主图指标源码公式_通达信指标公式_指标公式网...
  15. python倒计时弹框提示带注释_注意时间用python制作倒计时提醒工具
  16. Lightroom 2020年2月版新增功能
  17. 计算机图形管线(实时渲染管线)
  18. 分享一些实用的手机应用
  19. 游戏暴击(随机数和if判断)
  20. 44.网络安全渗透测试—[穷举篇7]—[网站会员批量登录穷举]

热门文章

  1. codesys【软PLC】
  2. java 分库分表框架 搭建_SpringBoot结合Sharding-JDBC实现分库分表
  3. 程序员也敢吃10元的盒饭
  4. python高手养成百家号_离骚是词吗 使用Python分析屈原《离骚》中高频词并生成词......
  5. 朴素贝叶斯算法(理论部分)-Theory of naive Bayes
  6. 在纺织企业中,MES项目管理能创造哪些价值?
  7. SQL Server2019字段类型说明等
  8. 真菜鸡的保研之路(从未设想之路)
  9. CentOS Linux下清理系统缓存
  10. 名画103 扬无咎《梅花图两幅》