java8新特性之Optional的深入解析

 更新时间:2019-02-06 22:00:12   作者:佚名   我要评论(0)

前言
最近脑袋发热追着java8源码看的很起劲,还有了执念,罪过。
本文以jdk1.8.0_111源码为例


public final class Optional<T> {}

Optional是一个为了解决

前言

最近脑袋发热追着java8源码看的很起劲,还有了执念,罪过。

本文以jdk1.8.0_111源码为例

public final class Optional<T> {}

Optional是一个为了解决NullPointerException设计而生可以包含对象也可以包含空的容器对象。封装了很多对空处理的方法也增加了filter、map这样的检索利器,其中函数式编程会有种炫酷到爆的感觉。

基础测试用例对象:

public class Java8OptionalTest {
 List<String> stringList = null;
 ICar car = new WeiLaiCar();
}

public class WeiLaiCar implements ICar {
 Integer wheels = new Integer(4);
}

Api中提供的4种optional

最核心的当属Optional对象,泛型的引入支持了所有对象类型,又增加对常用场景下的dubbo\int\long进行扩展。重点介绍一下Optional对象的方法其他三个类似。

  • public final class Optional<T> {
  • public final class OptionalDouble {
  • public final class OptionalInt {
  • public final class OptionalLong {

@FunctionalInterface
Predicate\Consumer\Supplier三个接口都是函数式接口

静态方法of

private Optional() {
 this.value = null;
}

构造方法被private,不能new但提供了of这样的静态方法去初始化类;

public static <T> Optional<T> of(T value) {
 return new Optional<>(value);
}
public static <T> Optional<T> ofNullable(T value) {
 return value == null ? empty() : of(value);
}
public static<T> Optional<T> empty() {
 @SuppressWarnings("unchecked")
 Optional<T> t = (Optional<T>) EMPTY;
 return t;
}

1、empty支持你去创建一个空的optional类,这样的类直接get()会报错:java.util.NoSuchElementException: No value present

2、of(x)传入的对象不能为null,而ofNullable(x)是支持传入null的对象,一般用这两个比较多。

present 方法

isPresent是用来判断optional中对象是否为null,ifPresent的参数是当对象不为null时执行的lamdba表达式。

public boolean isPresent() {
 return value != null;
}
public void ifPresent(Consumer<? super T> consumer) {
 if (value != null)
 consumer.accept(value);
}

示例详解介绍了ifPresent特性:

Java8OptionalTest test = new Java8OptionalTest();
Optional<Java8OptionalTest> optional = Optional.of(test);

pringTest(optional.isPresent());
//true
optional.ifPresent( a -> pringTest(a.getCar().getClass().getName()));
//com.ts.util.optional.WeiLaiCar
optional.ifPresent( a -> Optional.ofNullable(a.getStringList()).ifPresent(b -> pringTest("StringList:" + (b == null))));
//第一级的ifPresent是存在test对象,所以执行了lambda表达式,而第二级的ifPresent的stringList是null,所以没有执行表达式
optional.ifPresent( a -> Optional.ofNullable(a.getCar()).ifPresent(b -> pringTest("car:" + (b == null))));
//car:false
//第二级ifPresent的car对象是存在的,所以第二级的表达式执行了

map 方法

源码提供了两种map和flatMap。

  • map方法的参数是个当包含的对象不为null时才执行的lambda表达式,返回该表达式执行结果的封装optional对象,同理支持链式调用,逐层深入和递归递进很像;
  • flatMap区别在于lambda表达式的返回结果必须主动包裹Optinoal,否则报错
public<U> Optional<U> map(Function<? super T, ? extends U> mapper) {
 Objects.requireNonNull(mapper);
 if (!isPresent())
  return empty();
 else {
  return Optional.ofNullable(mapper.apply(value));
 }
}
public<U> Optional<U> flatMap(Function<? super T, Optional<U>> mapper) {
 Objects.requireNonNull(mapper);
 if (!isPresent())
  return empty();
 else {
  return Objects.requireNonNull(mapper.apply(value));
 }
}

测试示例:

Java8OptionalTest test = new Java8OptionalTest();
Optional<Java8OptionalTest> optional = Optional.of(test);

Optional opt1 = optional.map( a -> a.getCar());
pringTest(opt1.get());
//com.ts.util.optional.WeiLaiCar@5d6f64b1
int wheel = 0;//传统null判断写法
if(test != null){
 if(test.getCar() != null){//实际业务里面层级也许会超过3层
  wheel = test.getCar().getWheelCount();
 }
}
pringTest("传统:"+wheel);
//传统:4
Optional opt2 = optional.map( a -> a.getCar()).map(b -> b.getWheelCount());//Optional支持下的写法
pringTest("optinal:"+opt2.get());
//optinal:4
Optional opt3 = optional.map( a -> a.getStringList()).map(b -> b.size());
pringTest(opt3);
//Optional.empty

Optional opt4 = optional.flatMap(a -> Optional.of(a.getCar()));//主动包裹Optional对象
pringTest(opt4);
//Optional[com.ts.util.optional.WeiLaiCar@5d6f64b1]
Optional opt5 = optional.flatMap(a -> Optional.of(a.getCar())).flatMap(b -> Optional.ofNullable(b.getWheelCount()));
pringTest(opt5);
//Optional[4]

filter 方法

源码如下:

public Optional<T> filter(Predicate<? super T> predicate) {
 Objects.requireNonNull(predicate);
 if (!isPresent())
  return this;
 else
  return predicate.test(value) ? this : empty();
}

filter方法传入一个断言语句条件的lambda表达式,返回一个原对象的optional包装,所以支持链式调用;只要记住这三点你便掌握如何使用了。

看下面的例子:

Java8OptionalTest test = new Java8OptionalTest();

Optional<Java8OptionalTest> optional = Optional.of(test);

Optional result = optional.filter( a -> a.getCar() != null).filter( b -> b.getClass().getName() != null);
pringTest(result.isPresent()? result.get().getClass().getName(): result.isPresent());
//com.ts.util.Java8OptionalTest
Optional result1 = optional.filter( a -> a.getStringList() != null);
pringTest(result1.get());
//java.util.NoSuchElementException: No value present

orElse 方法

Api提供了三个方法。

  • orElse 当optional内对象为null就返回这个参数,比较像很多默认值设置;
  • orElseGet 基本同orElse,区别是传入参数支持lambda表达式,返回的就是表达式执行结果;
  • orElseThrow 也是传入lambda表达式,但是表达式是抛出异常
public T orElse(T other) {
 return value != null ? value : other;
}

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

public <X extends Throwable> T orElseThrow(Supplier<? extends X> exceptionSupplier) throws X {
 if (value != null) {
  return value;
 } else {
  throw exceptionSupplier.get();
 }
}

测试用例如下:

Java8OptionalTest one = null;
Java8OptionalTest test = new Java8OptionalTest();
Optional<Java8OptionalTest> optional = Optional.ofNullable(one);
pringTest(optional);
//Optional.empty
pringTest(optional.orElse(test));
//com.ts.util.Java8OptionalTest@5197848c
pringTest(optional.orElseGet(() -> new Java8OptionalTest()));
//com.ts.util.Java8OptionalTest@5d6f64b1
pringTest(optional.orElseThrow(() -> new RuntimeException("orElseThrow")));
//java.lang.RuntimeException: orElseThrow

总结

官方推出Optional绝不会就是替大家判断一下null,filter\map\orElse这三种使用场景是比较容易想到的,很多业务场景需要慢慢摸索使用。多函数式的用法需要好好掌握,技术发展是非常快速的。

后面会专门开一篇讲函数式和Lambda表达式用法,保持好奇心关注我的博客。

好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对脚本之家的支持。

您可能感兴趣的文章:

  • 利用Java8 Optional如何避免空指针异常详解
  • 使用Java8中Optional机制的正确姿势
  • JAVA8如何妙用Optional解决NPE问题详解
  • Java8中Optional的一些常见错误用法总结
  • java8中forkjoin和optional框架使用
  • Java8中Optional类型和Kotlin中可空类型的使用对比
  • Java8中新特性Optional、接口中默认方法和静态方法详解

相关文章

  • java8新特性之Optional的深入解析

    java8新特性之Optional的深入解析

    前言 最近脑袋发热追着java8源码看的很起劲,还有了执念,罪过。 本文以jdk1.8.0_111源码为例 public final class Optional<T> {} Optional是一个为了解决
    2019-02-06
  • PHP simplexml_load_string()函数实例讲解

    PHP simplexml_load_string()函数实例讲解

    PHP simplexml_load_string() 函数 实例 转换形式良好的 XML 字符串为 SimpleXMLElement 对象,然后输出对象的键和元素: <&#63;php $note=<<<XML <note>
    2019-02-03
  • PHP addslashes()函数讲解

    PHP addslashes()函数讲解

    PHP addslashes() 函数 实例 在每个双引号(")前添加反斜杠: <&#63;php $str = addslashes('What does "yolo" mean&#63;'); echo($str); &#63;> 定义
    2019-02-03
  • PHP addcslashes()函数讲解

    PHP addcslashes()函数讲解

    PHP addcslashes() 函数 实例 在字符 "W" 前添加反斜杠: <&#63;php $str = addcslashes("Hello World!","W"); echo($str); &#63;> 定义和用法 addcsla
    2019-02-03
  • python生成每日报表数据(Excel)并邮件发送的实例

    python生成每日报表数据(Excel)并邮件发送的实例

    逻辑比较简单 ,直接上代码 定时发送直接使用了win服务器的定时任务来定时执行脚本 #coding:utf-8 from __future__ import division import pymssql,sys,d
    2019-02-03
  • 使用Python快速制作可视化报表的方法

    使用Python快速制作可视化报表的方法

    我们可以试用可视化包——Pyechart。 Echarts是百度开源的一个数据可视化JS库,主要用于数据可视化。 pyecharts是一个用于生成Echarts图标的类库。实际就是Ec
    2019-02-03
  • PHP count()函数讲解

    PHP count()函数讲解

    PHP count() 函数 实例 计算 car 节点的子节点个数: <&#63;php $xml=<<<XML <cars> <car name="Volvo"> <child/> <child/> <child/> <child/> </ca
    2019-02-03
  • PHP simplexml_import_dom()函数讲解

    PHP simplexml_import_dom()函数讲解

    PHP simplexml_import_dom() 函数 实例 获取 DOM 文档节点并转换为 SimpleXML 节点: <&#63;php $dom=new domDocument; $dom->loadXML("<note><to>Tove</
    2019-02-03
  • 对python生成业务报表的实例详解

    对python生成业务报表的实例详解

    本文介绍一个用python结合xlsxwriter自动生成业务报表的程序。这里的业务数据采用的是指定的值,真实情况下需要其他程序来接入数据。 # -*- coding: utf-8
    2019-02-03
  • python生成带有表格的图片实例

    python生成带有表格的图片实例

    因为工作中需要,需要生成一个带表格的图片 例如: 直接在html中写一个table标签,然后单独把表格部分保存成图片 或者是直接将excel中的内容保存成一个图片 刚
    2019-02-03

最新评论