java8新特性之Optional类

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

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

一. 直观感觉

假设咱们须要返回一个字符串的长度,若是不借助第三方工具类,咱们须要调用str.length()方法:工具

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

若是采用Optional类,实现以下:this

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

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

二. 基本使用

1.对象建立

建立空对象设计

Optional<String> optStr = Optional.empty();

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

建立对象:不容许为空
Optional提供了方法of()用于建立非空对象,该方法要求传入的参数不能为空,不然抛NullPointException,示例以下:code

Optional<String> optStr = Optional.of(str);  // 当str为null的时候,将抛出NullPointException

建立对象:容许为空
若是不能肯定传入的参数是否存在null值的可能性,则能够用Optional的ofNullable()方法建立对象,若是入参为null,则建立一个空对象。示例以下:对象

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

2.流式处理

流式处理也是java8给咱们带来的一个重量级新特性,让咱们对集合的操做变得更加简洁和高效,下一篇关于java8新特性的文章,将对流失处理进行全面的讲解。这里Optional也提供了两个基本的流失处理:映射和过滤。blog

为了演示,咱们设计了一个User类,以下:

/**
 * @author: zhenchao.Wang 2016-9-24 15:36:56
 */
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));

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;  // 不能序列化

不过咱们能够采用以下替换策略:

private long phone;

public Optional<Long> getPhone() {
    return Optional.ofNullable(this.phone);
}
相关文章
相关标签/搜索