jdk8 optional 简介

 

    作为程序员,你肯定遇到过NullPointerException, 这个异常对于初出茅庐的新人, 还是久经江湖的老手都是不可避免的痛, 可又是那么的无能为力,为了解决它,你只能在使用某个值之前,对其进行判空处理。然而这样会使得代码变得臃肿不堪。幸好jdk8引入了optional来处理了null的问题,使得我们可以不再对null做过多的关心。

    先展示jdk8之前的写法

// First before jdk8
Long id = 0L;
User user = getUserById(id);
if (user != null) {
    String name = user.getName();
    System.out.println("name=" + name);
}

如果不对方法拿到的user 进行空判断,下面获取其属性很容易就会出现空指针异常。

jdk 8 写法

User userById = Optional.ofNullable(getUserById(id)).orElse(new User());
String name = userById.getName();
System.out.println("new name=" + name);

这种写法就避免了空的判断,代码显得很简便

下面介绍下 optional 的方法

1、首先是构造  of,  ofNullable, empty

// empty 构建空的optional对象
Optional empty = Optional.empty();

// of 构造user的optional 对象,user对象不为空, 如果为空,构建的时候就会报nullPointerException
User user = new User();
Optional userOptional = Optional.of(user);

// ofNull 构造optional对象,内部user如果为空,就构建空的optional对象
Optional userOptionalOfNull = Optional.ofNullable(user); 
   
Optional objectOptional = Optional.of(null);
System.out.println(objectOptional);


Exception in thread "main" java.lang.NullPointerException
	at java.util.Objects.requireNonNull(Objects.java:203)
	at java.util.Optional.(Optional.java:96)
	at java.util.Optional.of(Optional.java:108)
	at com.zbb.jdk.jdk8Test.optional.OptionalOfTest.main(OptionalOfTest.java:32) 
   

看下ofNullable 方法源码

jdk8 optional 简介_第1张图片

2、 获取 ifPresent, get, isPresent

Optional userOptional = Optional.ofNullable(getUserById(id));
// isPresent  判断optional对象是否存在,如果存在,返回true, 否则返回false
if(userOptional.isPresent()){
    // 已判断存在
    // get 如果创建的Optional中有值存在,则返回此值,否则抛出NoSuchElementException
    User user = userOptional.get();
    System.out.println("name" + user.getName());
}

// 创建的Optional中的值存在,则执行该方法的调用,否则什么也不做
// ifPresent方法的参数是一个函数式接口, 该方法无返回值,可以直接用lambda表达式
userOptional.ifPresent(user -> System.out.println("name=" + user.getName()));

最开始的例子也可以用isPresent来这样改,但是这样和原先判空并没有什么区别,只是方法不同而已,本质没区别,所以不推荐。看下这几个方法的源码

jdk8 optional 简介_第2张图片

3、获取 orElse, orElseGet, orElseThrow

Optional userOptional = Optional.ofNullable(getUserById(id));

//orElse 如果optional 有值就返回值,如果没有就返回一个默认值
// 默认值就是我们创建的一个类
User user = userOptional.orElse(new User("xiaohong", "123456789"));
System.out.println("name=" + user.getName()); // name=xiaohong

//orElseGet 如果optional有值就返回值,如果没有,就执行一个Supplier接口,返回生成的值
User userOrElseGet = userOptional.orElseGet(() -> new User("xiaohongGet", "123456789"));
System.out.println("name=" + userOrElseGet.getName()); //name=xiaohongGet

//orElseThrow 如果optional有值就返回值,如果没有,就返回一个指定的Supplier接口生成的异常
User userElseThrow = userOptional.orElseThrow(() -> new Exception("userOptional 为空!"));

// 为空时,抛出的异常
Exception in thread "main" java.lang.Exception: userOptional 为空!
	at com.zbb.jdk.jdk8Test.optional.OptionalOrElse.lambda$main$1(OptionalOrElse.java:29)
	at java.util.Optional.orElseThrow(Optional.java:290)
	at com.zbb.jdk.jdk8Test.optional.OptionalOrElse.main(OptionalOrElse.java:29)

看下源码

jdk8 optional 简介_第3张图片

orElseThrow 同理

4、过滤  filter

Optional userOptional = Optional.ofNullable(new User("xiao", "123456"));

//optional中的值符合条件,则返回该optional对象,否则返回空的optional对象
User user = userOptional.filter(u -> NAME.equals(u.getName())).orElse(new User("不满足", ""));
System.out.println("name=" + user.getName()); //name=xiao

jdk8 optional 简介_第4张图片

5、转化  map, flatMap

Optional optional = Optional.ofNullable("zhang,san");

//map  optional对象存在就执行Funciton函数
// 可以返回任意类型的值
// 该函数式接口对optional对象中的值进行修改,返回修改后的值
Optional optionalMap = optional.map(s -> s.split(",")); //Optional[[Ljava.lang.String;@19dfb72a]

// flatMap方法中的lambda表达式返回值必须是Optionl实例
Optional optionalFlatMap = optional.flatMap(s -> Optional.of("lisi"));//Optional[lisi]

jdk8 optional 简介_第5张图片

jdk8 optional 简介_第6张图片

看源码知道,map 和 flatmap 都是如果optional中的值存在,就对该值执行提供的Function函数调用,返回一个optional类型的值,否 
则就返回一个空的optional对象。

    最大的区别是map 函数执行之后是任意的类型数据,调用结束map 会再用optional对结果进行包装,而flatmap 是执行完之后就是一个optional 实例 ,flatmap 不会对结果进行optional封装。

转载于:https://my.oschina.net/u/2507440/blog/2960600

你可能感兴趣的:(jdk8 optional 简介)