java.util.Optional学习笔记

java.util.Optional是Java 8新增的类,作为一个持有实例的容器类,可以帮我们把判空的代码写得更优雅,并且该类还提供了一些实用的api,官方文档在这里,接下来我们通过实战来学习吧:

三种Optional构造方法

第一种. Optional.of(Object object):入参object不能为空,否则会抛出空指针异常,查看Optional源码发现会调用Objects.requireNonNull方法,里面有判空:

public static <T> T requireNonNull(T obj) {
        if (obj == null)
            throw new NullPointerException();
        return obj;
    }

第二种. Optional.ofNullable(Object object):入参object可以为空,如果object不为空,就创建一个Optional实例;如果object为空就返回一个static fainal的Option对象,注意这里不会新建Option实例,而是使用一个static final的实例EMPTY,这里比较有意思的是泛型的问题,例如我需要两个Optional对象,类型分别是String和Integer,代码如下:

Optional<String> optionalStr = Optional.ofNullable(null);
Optional<Integer> optionalInt = Optional.ofNullable(null);

类型不同又如何保证返回同一个对象呢?直接看ofNullable的源码,发现会调用empty方法:

public static<T> Optional<T> empty() {
        @SuppressWarnings("unchecked")
        Optional<T> t = (Optional<T>) EMPTY;
        return t;
    }

原来是通过强制转换实现的,再看EMPTY对象:

private static final Optional<?> EMPTY = new Optional<>();

是通过”?”声明的;

第三种. Optional.empty():就是上面分析Optional.ofNullable的时候用到的empty方法,直接返回一个static final的实例EMPTY;

Optional.of()方法的用法有点像断言,对象为空的时候代表着某种业务上不可接受的异常,需要尽早处理,并且业务拒绝执行,这种场景下可以使用Optional.of;

接下来我们开始实战吧;

例子中用到的对象:Student

Student是个普通的bean,有三个字段和对应的get&set方法

package com.bolingcavalry;

/**
 * @author willzhao
 * @version V1.0
 * @Description: 一个普通的bean
 * @email zq2599@gmail.com
 * @Date 2017/8/26 下午11:23
 */
public class Student {
    private int id;

    private String name;

    private int age;

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

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

Optional.ofNullable的用法

下面举例说明最常用的Optional.ofNullable,我们打算根据名称从其他系统获取student对象,如果对象为空就返回默认对象,先看不用Optional的时候我们平常是怎么写的,如下代码所示,标准的if&else判断:

    private Student queryById(int id){
        //TODO 这里模拟从数据库查询
        return null;
    }

    public Student getStudent(int id){
        Student student = queryById(id));

        //如果为空就返回DEFAULT对象
        return student==null ? DEFAULT : student;
    }

用Optional之后,如下所示,不需要通过判空来避免空指针异常了:

private Student queryById(int id){
        //TODO 这里模拟从数据库查询
        return null;
    }

    public Student getStudent(int id){
        Optional<Student> optional = Optional.ofNullable(queryById(id));

        //如果为空就返回DEFAULT对象
        return optional.orElse(DEFAULT);
    }

orElse方法可以指定一个value为空时的返回对象,如果这个对象需要调用方法才能获取(例如我们拿不到DEFAULT对象,要通过getDefault()方法才能拿到),这是就需要orElseGet方法来达到目的,如下:

private Student queryById(int id){
    //TODO 这里模拟从数据库查询
    return null;
}

private Student getDefault(){
    return DEFAULT;
}

public Student getStudent(int id){
    Optional<Student> optional = Optional.ofNullable(queryById(id));

    //如果为空就返回DEFAULT对象
    return optional.orElseGet(() -> getDefault());
}

Optional的map方法

假如我们的需求是student对象非空就返回name的大写,如果student对象为空就返回”invalid”,在没有Optional的时候写法如下,除了检查student变量是否为空,还要检查name是否为空:

private Student queryById(int id){
        //TODO 这里模拟从数据库查询
        return null;
}

public String getStudentUpperName(int id){
    Student student = queryById(id);
    if(student!=null && student.getName()!=null){
        return student.getName().toUpperCase();
    }

    return "invalid";
}

用了Optional可以这么写:

private Student queryById(int id){
    //TODO 这里模拟从数据库查询
    return null;
}

public String getStudentUpperName(int id){
    Optional<Student> optional = Optional.ofNullable(queryById(id));

    return optional.map(student -> student.getName())
                   .map(name -> name.toUpperCase())
                   .orElse("invalid");
}

由以上代码可以看到,map可以将一个Optional对象转换成另一个,第一次是将Optional转换成了Optional,第二次是将Optional转成了另一个Optional,只是这次将字符串换成了大写;

本次实战的源码已经上传到git上,地址是git@github.com:zq2599/blog_demos.git,里面有多个工程,本次用到的是optionaldemo,如下图红框所示:

以上就是Optional的基本用法,对Optional的使用是在习惯上对之前判空写法的挑战,但可以试着去习惯这个简单优雅的小工具;

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Java进阶之路

Java8新特性实践

18900
来自专栏pangguoming

JavaScript的IIFE(即时执行方法)

前面的话   严格来讲,IIFE并不是闭包,因为它并不满足函数成为闭包的三个条件。但一般地,人们认为IIFE就是闭包,毕竟闭包有多个定义。本文将详细介绍IIFE...

32550
来自专栏撸码那些事

C#集合类型大揭秘

18240
来自专栏技术小黑屋

Java中的字符串常量池

Java中字符串对象创建有两种形式,一种为字面量形式,如String str = "droid";,另一种就是使用new这种标准的构造对象的方法,如String...

20720
来自专栏老九学堂

干货 | Java 中不得不知的异常和处理详解

简介 程序运行时,发生的不被期望的事件,它阻止了程序按照程序员的预期正常执行,这就是异常。异常发生时,是任程序自生自灭,立刻退出终止,还是输出错误给用户?或者用...

41170
来自专栏猿人谷

C++ primer里的template用法

template 的用法     在程序设计当中经常会出现使用同种数据结构的不同实例的情况。例如:在一个程序中     可以使用多个队列、树、图等结构来组织数据...

23150
来自专栏对角另一面

读Zepto源码之操作DOM

这篇依然是跟 dom 相关的方法,侧重点是操作 dom 的方法。 读Zepto源码系列文章已经放到了github上,欢迎star: reading-zepto ...

23800
来自专栏chenjx85的技术专栏

leetcode-697-Degree of an Array

26540
来自专栏java学习

面试题55(考察求职者对类的声明的掌握)

(不定项选择题)在Java中下面Class的声明哪些是错误的? A public abstract final class Test { abstract ...

35450
来自专栏Python小屋

Python实现单链表

class Node: '''节点结构''' def __init__(self, data, nextNode=None): #设置当前节点的值和指向下...

37350

扫码关注云+社区

领取腾讯云代金券