Java 8th 函数式编程:Optional 类型

朝朝暮暮
• 阅读 1738

NullPointException 可以说是所有 java 程序员都遇到过的一个异常,虽然 java 从设计之初就力图让程序员脱离指针的苦海,但是指针确实是实际存在的,而 java 设计者也只能是让指针在 java 语言中变得更加简单、易用,而不能完全的将其剔除,所以才有了我们日常所见到的关键字 null

空指针异常是一个运行时异常,对于这一类异常,如果没有明确的处理策略,那么最佳实践在于让程序早点挂掉,但是很多场景下不是开发人员没有具体的处理策略,而是根本没有意识到空指针异常的存在。当异常真的发生的时候,处理策略也很简单,在存在异常的地方添加一个 if 语句判定即可,但是这样的应对策略会让我们的程序出现越来越多的 null 判定。一个良好的程序设计应该让代码中尽量少出现 null 关键字,而 8th 所提供的 Optional 类则在减少 NullPointException 的同时,也提升了代码的美观度。但首先我们需要明确的是它并 不是对 null 关键字的替代策略,而是对于 null 判定提供了一种更加优雅的实现,从而尽可能地避免 NullPointException

下面通过一个小示例直观感受一下,假设我们需要返回一个字符串的长度,如果不借助第三方工具类,我们需要调用 str.length() 方法:

if(null == str) { // 空指针判定
    return 0;
}
return str.length();

如果采用 Optional 类,实现如下:

return Optional.ofNullable(str).map(String::length).orElse(0);

Optional 的代码相对更加简洁,当代码量较大时,我们很容易忘记进行 null 判定,但是使用 Optional 类则会避免这类问题。

一. 基本使用

1.1 Optional 对象的创建

  • 创建空对象
Optional<String> optStr = Optional.empty();

上面的示例代码调用 empty() 方法创建了一个空的 Optional<String> 对象型。

  • 创建对象:不允许我空

Optional 提供了方法 of() 用于创建非空对象,该方法要求传入的参数不能为空,否则抛 NullPointException,示例如下:

Optional<String> optStr = Optional.of(str);  // 当str为null的时候,将抛出NullPointException
  • 创建对象:允许为空

如果不能确定传入的参数是否存在 null 值的可能性,则可以用 Optional 的 ofNullable() 方法创建对象,如果入参为 null 则创建一个空对象。示例如下:

Optional<String> optStr = Optional.ofNullable(str);  // 如果str是null,则创建一个空对象

1.2 流式数据处理

流式数据处理也是 8th 给我们带来的一个重量级新特性,让我们对集合的操作变得更加简洁和高效,本系列下一篇将对流式数据处理进行全面的讲解。Optional 类也提供了两个基本的流失处理:映射和过滤。

为了演示,我们设计了一个 User 类,如下:

public class User {
    private long id;
    private String name;
    private int age;
    private Optional<Long> phone;
    private Optional<String> email;

    public User(String name, int age) {
        this.name = name;
        this.age = age;
    }

    // 省略setter和getter
}

手机和邮箱不是一个人的必须有的,所以我们利用 Optional 类定义。

  • 映射:map 与 flatMap

映射是将输入转换成另外一种形式的输出的操作,比如前面例子中我们输入字符串,而输出的是字符串的长度,这就是一种映射,我们利用方法 map() 进行实现。假设我们希望获得一个人的姓名,我们可以如下实现:

String name = Optional.ofNullable(user).map(User::getName).orElse("no name");

这样当入参 user 不为空的时候则返回其 name,否则返回 no name。如我我们希望通过上面方式得到 phone 或 email,利用上面的方式则行不通了,因为 map 之后返回的是 Optional,我们把这种称为 Optional 嵌套,我们必须再 map 一次才能拿到我们想要的结果:

long phone = optUser.map(User::getPhone).map(Optional::get).orElse(-1L);

其实这个时候更好的方式是利用 flatMap,一步拿到我们想要的结果:

long phone = optUser.flatMap(User::getPhone).orElse(-1L);

flapMap 可以将方法返回的各个流扁平化成为一个流,具体在下一篇专门讲流式数据处理的文章中细说。

  • 过滤:fliter

filiter,顾名思义是过滤的操作,我们可以将过滤操作做为参数传递给该方法以实现过滤目的,假如我们希望筛选 18 周岁以上的成年人,则可以实现如下:

optUser.filter(u -> u.getAge() >= 18).ifPresent(u -> System.out.println("Adult:" + u));

1.3 默认行为

默认行为是当 Optional 在不满足条件时所执行的操作,比如在上面的例子中我们使用的 orElse() 就是一个默认操作,用于在 Optional 对象为空时执行特定操作,当然也有一些默认操作是当满足条件的对象存在时执行的操作。

  • get()

get 方法用于获取变量的值,但是当变量不存在时则会抛出 NoSuchElementException,所以如果不确定变量是否存在则不建议使用

  • orElse(T other)

当 Optional 的变量不满足给定条件时,则执行 orElse,比如前面当 str 为 null 时返回 0。

  • orElseGet(Supplier<? extends X> expectionSupplier)

如果条件不成立时需要执行相对复杂的逻辑而不是简单的返回操作,则可以使用 orElseGet 实现:

long phone = optUser.map(User::getPhone).map(Optional::get).orElseGet(() -> {
    // do something here
    return -1L;
});
  • orElseThrow(Supplier<? extends X> expectionSupplier)

get() 方法类似,都是在不满足条件时返回异常,不过这里我们可以指定返回的异常类型。

  • ifPresent(Consumer<? super T>)

当满足条件时执行传入的参数化操作。

二. 注意事项

Optional 是一个 final 类且未实现任何接口,所以当我们在利用该类包装定义类的属性的时候,如果我们定义的类有序列化的需求,那么因为 Optional 没有实现 Serializable 接口,这个时候执行序列化操作就会有问题:

public class User implements Serializable {
    private long id;
    private String name;
    private int age;
    private Optional<Long> phone;  // 不能序列化
    private Optional<String> email;  // 不能序列化
}

不过我们可以采用如下替换策略 Optinal:

private long phone;

public Optional<Long> getPhone() {
    return Optional.ofNullable(this.phone);
}

看来 Optional 类在设计的时候就没有考虑将它作为类的字段使用。

最后提醒一点,Optional 好用但不能滥用,在设计一个接口方法时是否采取 Optional 类型返回需要斟酌,一味的使用会让代码变得比较啰嗦,反而破坏了代码的简洁性。


鉴于作者水平有限,文中不免有错误之处,欢迎批评指正
个人博客:www.zhenchao.org

点赞
收藏
评论区
推荐文章
Wesley13 Wesley13
4年前
java,poi读取excel单元格为空的数据,出现空指针异常:java.lang.NullPointerException
java,poi读取excel单元格为空的数据,出现空指针异常:java.lang.NullPointerException参考文章:(1)java,poi读取excel单元格为空的数据,出现空指针异常:java.lang.NullPointerException(https://www.oschina.net/action/GoToLi
Wesley13 Wesley13
4年前
jdk8 optional
Optional是jdk1.8(新命名方式应该说JDK8)推出的解决空指针问题的类,解决空指针这个说法不是准确,应该说是方便简洁处理空指针问题吧。毕竟jdk8推出了很多新语言使java语言编写程序更加简洁。参考:https://www.jianshu.com/p/d81a5f7c9c4e(https://www.oschina.net/action
Wesley13 Wesley13
4年前
java常见问题整理
1.什么是Java虚拟机?为什么Java被称作是“平台无关的编程语言”?Java虚拟机是一个可以执行Java字节码的虚拟机进程。Java源文件被编译成能被Java虚拟机执行的字节码文件。Java被设计成允许应用程序可以运行在任意的平台,而不需要程序员为每一个平台单独重写或者是重新编译。Java虚拟机让这个变为可能,因为它知道底层硬件平台的指令
Wesley13 Wesley13
4年前
Java四种引用类型
引用与对象每种编程语言都有自己操作内存中元素的方式,例如在C和C里是通过指针,而在Java中则是通过“引用”。在Java中一切都被视为了对象,但是我们操作的标识符实际上是对象的一个引用(reference)。//创建一个引用,引用可以独立存在,并不一定需要与一个对象关联Strings;
Wesley13 Wesley13
4年前
Java 概述
Java概述Java编程语言是一种通用的、并行的、基于类的、面向对象的语言。它被设计得非常简单,这样程序员可以在该语言上流畅的交流。Java编程语言与C和C有关联,但组织却截然不同,其中也省略了其他语言的一些用法,比如指针。它的目的是作为一个生产性语言,而不是一个研究性语言,因此,在设计上避免了包括新的和未经考验的功能。
Wesley13 Wesley13
4年前
Java8(5):使用 Optional 处理 null
Java8(5):使用Optional处理null写过Java程序的同学,一般都遇到过NullPointerException:)——为了不抛出这个异常,我们便会写如下的代码:UserusergetUserById(id);if(user!
Wesley13 Wesley13
4年前
Java有关于Null的九件事
为什么在Java中需要学习null?因为如果你对null不注意,Java将使你遭受空指针异常的痛苦,并且你也会得到一个沉痛的教训。精力充沛的编程是一门艺术,你的团队、客户和用户将会更加欣赏你。以我的经验来看,导致空指针异常的一个最主要的原因是对Java中null的知识还不够。你们当中的很多已经对null很熟悉了,但是对那些不是很熟悉的来说,可以学到一些关于n
Wesley13 Wesley13
4年前
Java线程 Top 50
不管你是新程序员还是老手,你一定在面试中遇到过有关线程的问题。Java语言一个重要的特点就是内置了对并发的支持,让Java大受企业和程序员的欢迎。大多数待遇丰厚的Java开发职位都要求开发者精通多线程技术并且有丰富的Java程序开发、调试、优化经验,所以线程相关的问题在面试中经常会被提到。在典型的Java面试中,面试官会从线程的基本概念问起,如:为什
Stella981 Stella981
4年前
Kotlin代码检查在美团的探索与实践
背景Kotlin有着诸多的特性,比如空指针安全、方法扩展、支持函数式编程、丰富的语法糖等。这些特性使得Kotlin的代码比Java简洁优雅许多,提高了代码的可读性和可维护性,节省了开发时间,提高了开发效率。这也是我们团队转向Kotlin的原因,但是在实际的使用过程中,我们发现看似写法简单的Kotlin代码,可能隐藏着不容忽视的额外开销。本文剖析了K
Wesley13 Wesley13
4年前
Java学习路线图
JavaJava是一门面向对象编程语言,不仅吸收了C语言的各种优点,还摒弃了C里难以理解的多继承、指针等概念,因此Java语言具有功能强大和简单易用两个特征。Java语言作为静态面向对象编程语言的代表,极好地实现了面向对象理论,允许程序员以优雅的思维方式进行复杂的编程。Java具有简单性、面向对象、分布式、健壮性、安全性、平台独立
Wesley13 Wesley13
4年前
Java是一门面向对象编程语言的理解
Java是一门面向对象编程语言。不仅吸收了C语言的各种优点,还摒弃了C里难以理解的多继承、指针等概念,因此Java语言具有功能强大和简单易用两个特征。Java语言作为静态面向对象编程语言的代表,极好地实现了面向对象理论,允许程序员以优雅的思维方式进行复杂的编程。Java具有:简单性面向对象分布式健壮性安全性
朝朝暮暮
朝朝暮暮
Lv1
从此带我徜徉你骨肉里钟情的汉水辽河,有何不可
文章
3
粉丝
0
获赞
0