NullPointerException相信每个JAVA程序员都不陌生,是JAVA应用程序中最常见的异常。之前,Google Guava项目曾提出用Optional类来包装对象从而解决NullPointerException。受此影响,JDK8的类中也引入了Optional类,在新版的SpringData Jpa和Spring Redis Data中都已实现了对该方法的支持。

1、Optional类

 /**
* A container object which may or may not contain a non-null value.
* If a value is present, {@code isPresent()} will return {@code true} and
* {@code get()} will return the value.
*
* @since 1.8
*/
public final class Optional<T> {
/**
* Common instance for {@code empty()}.
*/
private static final Optional<?> EMPTY = new Optional<>(); /**
* If non-null, the value; if null, indicates no value is present
*/
private final T value; // 其他省略
}

  该方法的注释大致意思是:Optional是一个容器对象,它可能包含空值,也可能包含非空值。当属性value被设置时,isPesent()方法将返回true,并且get()方法将返回这个值。

  该类支持泛型,即其属性value可以是任何对象的实例。

2、Optional类的方法

    序号    

方法

方法说明
1
private Optional()
 无参构造,构造一个空Optional
2
private Optional(T value)
 根据传入的非空value构建Optional
3
public static<T> Optional<T> empty()
返回一个空的Optional,该实例的value为空
4

public static <T> Optional<T> of(T value)
根据传入的非空value构建Optional,与Optional(T value)方法作用相同
5

public static <T> Optional<T> ofNullable(T value)

与of(T value)方法不同的是,ofNullable(T value)允许你传入一个空的value,

当传入的是空值时其创建一个空Optional,当传入的value非空时,与of()作用相同

6

public T get()
 返回Optional的值,如果容器为空,则抛出NoSuchElementException异常
7

public boolean isPresent()
 判断当家Optional是否已设置了值
8

public void ifPresent(Consumer<? super T> consumer)
 判断当家Optional是否已设置了值,如果有值,则调用Consumer函数式接口进行处理
9

public Optional<T> filter(Predicate<? super T> predicate)
 如果设置了值,且满足Predicate的判断条件,则返回该Optional,否则返回一个空的Optional
10

public<U> Optional<U> map(Function<? super T, ? extends U> mapper)

如果Optional设置了value,则调用Function对值进行处理,并返回包含处理后值的Optional,否则返回空Optional

11

public<U> Optional<U> flatMap(Function<? super T, Optional<U>> mapper)
 与map()方法类型,不同的是它的mapper结果已经是一个Optional,不需要再对结果进行包装
12

public T orElse(T other)
 如果Optional值不为空,则返回该值,否则返回other 
13
public T orElseGet(Supplier<? extends T> other)
如果Optional值不为空,则返回该值,否则根据other另外生成一个
14
public <X extends Throwable> T orElseThrow(Supplier<? extends X> exceptionSupplier)
throws X
如果Optional值不为空,则返回该值,否则通过supplier抛出一个异常

3、Optional类的方法举例

  《Java 8 Features Tutorial – The ULTIMATE Guide》中给出了两个简单的例子,我们从这两个例子入手来简单了解一下Optional容器的使用。

  示例一:

 public static void main(String[] args) {
2   Optional< String > fullName = Optional.ofNullable( null );
3   System.out.println( "Full Name is set? " + fullName.isPresent() );
4   System.out.println( "Full Name: " + fullName.orElseGet( () -> "[none]" ) );
5   System.out.println( fullName.map( s -> "Hey " + s + "!" ).orElse( "Hey Stranger!" ) );
6 }

  运行结果:

Full Name is set? false
Full Name: [none]
Hey Stranger!  

  说明:

  ifPresent()方法当Optional实例的值非空时返回true,否则返回false;

  orElseGet()方法当Optional包含非空值时返回该值,否则通过接收的function生成一个默认的;

  map()方法转换当前Optional的值,并返回一个新的Optional实例;

  orElse()方法与orElseGet方法相似,不同的是orElse()直接返回传入的默认值。

  示例二:修改示例一,使其生成一个非空值的Optional实例

 Optional< String > firstName = Optional.of( "Tom" );
System.out.println( "First Name is set? " + firstName.isPresent() );
System.out.println( "First Name: " + firstName.orElseGet( () -> "[none]" ) );
System.out.println( firstName.map( s -> "Hey " + s + "!" ).orElse( "Hey Stranger!" ) );

  输出结果:

First Name is set? true
First Name: Tom
Hey Tom!

  可以清晰地看出与示例一的区别。这不但简洁了我们的代码,而且使我们的代码更便于阅读。

  下面看一下例子中使用到的几个方法的源码:

  1)、of

 public static <T> Optional<T> of(T value) {
return new Optional<>(value);
}

  2)、isPresent

 public boolean isPresent() {
return value != null;
}

  3)、orElseGet

 public T orElseGet(Supplier<? extends T> other) {
return value != null ? value : other.get();
}

  4)、orElse

 public T orElse(T other) {
return value != null ? value : other;
}

  其他方法源码,读者可以去Optional源码中查看。

4、使用Optional避免空指针

  在我们日常开发过程中不可避免地会遇到空指针问题,在以前,出现空指针问题,我们通常需要进行调试等方式才能最终定位到具体位置,尤其是在分布式系统服务之间的调用,问题更难定位。在使用Optional后,我们可以将接受到的参数对象进行包装,比如,订单服务要调用商品服务的一个接口,并将商品信息通过参数传入,这时候,传入的商品参数可能直接传入的就是null,这时,商品方法可以使用Optional.of(T)对传入的对象进行包装,如果T为空,则会直接抛出空指针异常,我们看到异常信息就能立即知道发生空指针的原因是参数T为空;或者,当传入的参数为空时,我们可以使用Optional.orElse()或Optional.orElseGet()方法生成一个默认的实例,再进行后续的操作。

  下面再看个具体例子:在User类中有个Address类,在Address类中有个Street类,Street类中有streetName属性,现在的需求是:根据传入的User实例,获取对应的streetName,如果User为null或Address为null或Street为null,返回“nothing found”,否则返回对应的streetName。

  实现一:

 @Data
public class User {
private String name;
private Integer age;
private Address address;
}
 @Data
public class Address {
private Street street;
}
 @Data
public class Street {
private String streetName;
private Integer streetNo;
}
 public String getUserSteetName(User user) {

     if(null != user) {

         Address address = user.getAddress();

         if(null != address) {

             Street street = address.getStreet();

             if(null != street) {
return street.getStreetName();
}
}
} return "nothing found";
}

  实现二,使用Optional:

  在实现一中明显的问题是if判断层级太深,下面复用Optional改写:

 @Data
public class User {
private String name;
private Integer age;
private Optional<Address> address = Optional.empty();
}
 @Data
public class Address {
private Optional<Street> street = Optional.empty();
}
 @Data
public class Street {
private String streetName;
private Integer streetNo;
}
 public String getUserSteetName(User user) {

     Optional<User> userOptional = Optional.ofNullable(user);
final String streetName = userOptional.orElse(new User()).getAddress().orElse(new Address()).getStreet().orElse(new Street()).getStreetName();
return StringUtils.isEmpty(streetName) ? "nothing found" : streetName;
}

  利用orElse()方法给定默认值的方式确保不会报空指针问题问题,同时也能实现需求。

Java8新特性之五:Optional的更多相关文章

  1. 乐字节-Java8新特性之Optional

    上一篇小乐带大家了解了Java新特性之Stream,接下来将会继续述说Java新特性之Optional Optional<T>类(java.util.Optional)是一个容器类,代表一 ...

  2. 【Java8新特性】Optional类在处理空值判断场景的应用 回避空指针异常 编写健壮的应用程序

    一.序言 空值异常是应用运行时常见的异常,传统方式为了编写健壮的应用,常常使用多层嵌套逻辑判断回避空指针异常.Java8新特性之Optional为此类问题提供了优雅的解决方式. 广大程序员朋友对空值异 ...

  3. java8新特性之Optional类

    NullPointException可以说是所有java程序员都遇到过的一个异常,虽然java从设计之初就力图让程序员脱离指针的苦海,但是指针确实是实际存在的,而java设计者也只能是让指针在java ...

  4. 【Java8新特性】- Optional应用

    Java8新特性 - Optional应用 生命不息,写作不止 继续踏上学习之路,学之分享笔记 总有一天我也能像各位大佬一样 一个有梦有戏的人 @怒放吧德德 分享学习心得,欢迎指正,大家一起学习成长! ...

  5. java8新特性六-Optional 类

    Optional 类是一个可以为null的容器对象.如果值存在则isPresent()方法会返回true,调用get()方法会返回该对象. Optional 是个容器:它可以保存类型T的值,或者仅仅保 ...

  6. Java8新特性之Optional,如何优雅地处理空指针

    是什么 ​ 从 Java 8 引入的一个很有趣的特性是 Optional 类.Optional 类主要解决的问题是臭名昭著的空指针异常(NullPointerException)-- 每个 Java ...

  7. Java8新特性之Optional

    空指针异常一直是困扰Java程序员的问题,也是我们必须要考虑的.当业务代码中充满了if else判断null 的时候程序变得不再优雅,在Java8中提供了Optional类为我们解决NullPoint ...

  8. 【Java8新特性】Optional 类

    概述 Optional 类是一个可以为null的容器对象.如果值存在则isPresent()方法会返回true,调用get()方法会返回该对象. Optional 是个容器:它可以保存类型T的值,或者 ...

  9. 乐字节-Java8新特性之Date API

    上一篇文章,小乐给大家带来了Java8新特性之Optional,接下来本文将会给大家介绍Java8新特性之Date API 前言: Java 8通过发布新的Date-Time API来进一步加强对日期 ...

随机推荐

  1. sudo apt-get 与 yum安装有啥区别

    rpm包和deb包是两种Linux系统下最常见的安装包格式,在安装一些软件或服务的时候免不了要和它们打交道. rpm包主要应用在RedHat系列包括 Fedora等发行版的Linux系统上, deb包 ...

  2. Python_CRC32

    #计算文件的CRC32值 import sys import zlib import os.path filename=sys.argv[1] print(filename) if os.path.i ...

  3. codeforces 979D Kuro and GCD and XOR and SUM

    题意: 给出两种操作: 1.添加一个数字x到数组. 2.给出s,x,k,从数组中找出一个数v满足gcd(x,k) % v == 0 && x + v <= s && ...

  4. linux常用的时间获取函数(time,gettimeofday,clock_gettime,_ftime,localtime,strftime )

    time()提供了秒级的精确度 1.头文件 <time.h> 2.函数原型 time_t time(time_t * timer) 函数返回从TC1970-1-1 0:0:0开始到现在的秒 ...

  5. c++右值引用以及使用

    前几天看了一篇文章<4行代码看看右值引用> 觉得写得不错,但是觉得右值引用的内容还有很多可以去挖掘学习,所以总结了一下,希望能对右值引用有一个更加深层次的认识 一.几个基本概念 1.1左值 ...

  6. 深入Spring Boot:那些注入不了的Spring占位符(${}表达式)

    Spring里的占位符 spring里的占位符通常表现的形式是: <bean id="dataSource" destroy-method="close" ...

  7. Java 并行与并发

    Java 并行与并发 注意两个词:并行(Concurrent) 并发(Parallel) 并行:是逻辑上同时发生,指在某一个时间内同时运行多个程序 并发:是物理上同时发生,指在某一个时间点同时运行多个 ...

  8. windows命令中的cd

    cd命令的作用为改变文件夹,也就是跳转目录.切换路径的意思.它后面可以接驱动器符号.完整路径和相对路径. 打开命令行窗口的时候,默认的目录位于当前用户所在的路径下,比如:C:\Users\koi\De ...

  9. Java Script 学习笔记 (二) Casper JS

    1. click() VS mouse.click() 在写自动化脚本要勾选一个复选框时,用casper.mouse.click() 无法选上这个checkbox, 需要用到casper.click( ...

  10. Go 1.9 sync.Map揭秘

    Go 1.9 sync.Map揭秘 目录 [−] 有并发问题的map Go 1.9之前的解决方案 sync.Map Load Store Delete Range sync.Map的性能 其它 在Go ...