java8新特性学习四(Optional类)

Posted 马宝云

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了java8新特性学习四(Optional类)相关的知识,希望对你有一定的参考价值。

在开发中,我们调用一个方法得到了一个对象返回值,却不能直接取对象返回值的属性。我们首先要判断这个对象返回值是否为null,只有在非空的前提下才能获取其属性值,不然直接取容易出现空指针异常。

而Java 8引入了一个新的Optional类。

Option<T>类(java.util.Optional)是一个容器类,代表一个值存在或不存在,原来用null表示一个值不存在,现在用Optional可以更好的表达这个概念。并且可以避免空指针异常。Optional类的javadoc描述如下:这是一个可以为null的容器对象。如果值存在则isPresent()方法会返回true,调用get()方法会返回该对象。

Optional提供很多有用的方法,这样我们就不用显式进行空值检测。

创建Optional类对象的方法:

  Optional.of(T t):通过工厂方法创建Optional类。需要注意的是,创建对象时传入的参数不能为null。如果传入参数为null,则抛出NullPointerException 。

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

  Optional.empty():创建一个空的Optional实例;在空的Optional实例上调用get(),抛出NoSuchElementException

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

  Optional.ofNullable(T t):为指定的值创建一个Optional,如果指定的值为null,则返回一个空的Optional。ofNullable与of方法相似,唯一的区别是可以接受参数为null的情况。

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

判断Optional容器中是否包含对象:

  boolean isPresent():判断是否包含对象;如果值存在返回true,否则返回false

  void ifPresent(Consumer<? super T> consumer):如果有值,就执行Consumer接口的实现代码,并且该值会作为参数传递给它。

  //如果入参为null,则创建一个空对象
        Optional<String> optStr3=Optional.ofNullable(str);
        System.out.println(optStr3);
        optStr3.ifPresent((value)->{
            System.out.println("The length of the value is: " + value.length());
        });

映射:

map(Function f):

如果有值,则对其执行调用mapping函数得到返回值。如果返回值不为null,则创建包含mapping返回值的Optional作为map方法返回值,否则返回空Optional。

map方法用来对Optional实例的值执行一系列操作。通过一组实现了Function接口的lambda表达式传入操作。

  @Test
    public void test1(){
      //map方法执行传入的lambda表达式参数对Optional实例的值进行修改。
      //为lambda表达式的返回值创建新的Optional实例作为map方法的返回值。
        String str="aabbcc";
        Optional<String> name=Optional.of(str);
        Optional<String> upperName = name.map((value) -> value.toUpperCase());
        System.out.println(upperName.orElse("No value found"));
    }

flatMap(Function mapper):

如果有值,为其执行mapping函数返回Optional类型返回值,否则返回空Optional。flatMap与map(Funtion)方法类似,区别在于flatMap中的mapper返回值必须是Optional。调用结束时,flatMap不会对结果用Optional封装。

flatMap方法与map方法类似,区别在于mapping函数的返回值不同。map方法的mapping函数返回值可以是任何类型T,而flatMap方法的mapping函数必须是Optional。

 @Test
    public void test2(){
      //flatMap与map(Function)非常类似,区别在于传入方法的lambda表达式的返回类型。
      //map方法中的lambda表达式返回值可以是任意类型,在map函数返回之前会包装为Optional。
      //但flatMap方法中的lambda表达式返回值必须是Optionl实例。
        String str="aabbcc";
        Optional<String> name=Optional.of(str);
        Optional<String> upperName = name.flatMap((value) -> Optional.of(value.toUpperCase()));
        System.out.println(upperName.orElse("No value found"));
    }

获取Optional容器的对象:

  T get():如果Optional有值则将其返回,否则抛出NoSuchElementException。

  T orElse(T other):如果有值则将其返回,否则返回指定的other对象

  @Test
    public void test4(){
        //如果有值orElse方法会返回Optional实例,否则返回传入的错误信息。
        Optional<String> name = Optional.of("aaabbcc");
        //创建没有值的Optional实例,例如值为‘null‘
        Optional empty = Optional.ofNullable(null);

        System.out.println(name.orElse("There is some value!"));
        System.out.println(empty.orElse("There is no value present!"));
    }

  T orElseGet(Supplier<? extends T> other):如果有值则将其返回,否则返回由Supplier接口实现提供的对象。

 @Test
    public void test5(){
        //如果有值orElse方法会返回Optional实例,否则返回传入的错误信息。
        Optional<String> name = Optional.of("aaabbcc");
        //创建没有值的Optional实例,例如值为‘null‘
        Optional empty = Optional.ofNullable(null);

        //orElseGet与orElse类似,区别在于传入的默认值。
        //orElseGet接受lambda表达式生成默认值。
        System.out.println(name.orElseGet(() -> "Default Value"));
        System.out.println(empty.orElseGet(() -> "Default Value"));
    }

  T orElseThrow(Supplier<? extends X> exceptionSupplier):如果有值则将其返回,否则抛出由Supplier接口实现提供的异常。

 @Test
    public void test6(){
        //如果有值orElse方法会返回Optional实例,否则返回传入的错误信息。
        Optional<String> name = Optional.of("aaabbcc");
        //创建没有值的Optional实例,例如值为‘null‘
        Optional empty = Optional.ofNullable(null);

        //orElseThrow与orElse方法类似,区别在于返回值。
        //orElseThrow抛出由传入的lambda表达式/方法生成异常。
        try {
           // name.orElseThrow(Exception::new);
           // empty.orElseThrow(Exception::new);
            empty.orElseThrow(new Supplier() {
                @Override
                public Object get() {
                    return new Exception();
                }
            });
        } catch (Throwable throwable) {
            throwable.printStackTrace();
        }
    }

 filter

filter个方法通过传入限定条件对Optional实例的值进行过滤。文档描述如下:如果有值并且满足断言条件返回包含该值的Optional,否则返回空Optional。这里可以传入一个lambda表达式。对于filter函数我们应该传入实现了Predicate接口的lambda表达式。

 @Test
    public void test3(){
        //filter方法检查给定的Option值是否满足某些条件。
        //如果满足则返回同一个Option实例,否则返回空Optional。
        String str="aabbccd";
        Optional<String> name=Optional.of(str);
        Optional<String> longName = name.filter((value) -> value.length() > 6);
        System.out.println(longName.orElse("The name is less than 6 characters"));//输出Sanaulla

        //另一个例子是Optional值不满足filter指定的条件。
        Optional<String> anotherName = Optional.of("Sana");
        Optional<String> shortName = anotherName.filter((value) -> value.length() > 6);
        //输出:name长度不足6字符
        System.out.println(shortName.orElse("The name is less than 6 characters"));
    }

 

 

 

以上是关于java8新特性学习四(Optional类)的主要内容,如果未能解决你的问题,请参考以下文章

java8新特性学习

java8新特性学习

java8新特性学习五(接口中的默认方法与静态方法)

java8新特性学习五(接口中的默认方法与静态方法)

java8新特性学习:stream与lambda

java8新特性学习六(新时间日期API)