一、前言

在我们编程过程中因为疏忽经常会遇到NullPointerException空指针异常,所以与此同时就避免不了对变量一层一层的进行判空。JDK8新特性所提供的Optional类,可以让我们更优雅的进行判空操作。学习其的意义就是提醒开发者注意空值情况,并在一定程度上优雅处理问题。

例如下图传统判空:

lua判断集合是否包含一个值_lua判断集合是否包含一个值


如上图,一旦代码量大起来了,条件多了,代码就会变得很冗余,变得难以维护,通过使用Optional就可以很好的处理,让我们继续了解。.

Optional 类是一个可以为null的容器对象。如果值存在则isPresent()方法会返回true,调用get()方法会返回该对象。Optional 是个容器:它可以保存类型T的值,或者仅仅保存null。Optional提供很多有用的方法,这样我们就不用显式进行空值检测。

二、Optional API介绍

首先了解一下Optional为我们提供使用的方法:

lua判断集合是否包含一个值_User_02


lua判断集合是否包含一个值_属性值_03

三、Optional实战

1、对象操作

Optional.of()和Optional.ofNullable()用法相似。
Optional.of()创建对象时传入的参数不能为null。如果传入参数为null,则抛出NullPointerException,而Optional.ofNullable()允许传null。
后续以Optional.ofNullable()为例,展示使用optional和不使用的不同写法

1.1 ifPresent(不为null则运行函数体)

如果user不为null则调用其他的方法或操作,例如:

User user = userService.getById("123456");
 //未使用Optional
  if (user != null){
  	   //执行user不为null的逻辑
       System.err.println(user.getUserName());
   }
// 使用Optional
 Optional.ofNullable(user).ifPresent(u -> System.out.printf(u.getUserName()));

1.2 filter(条件过滤)

User user = userService.getById("02BEC97B_4CE3_5E6B_7A4C_B03A26D93A9E");
    		//未使用Optional
    if (user != null){
        if (user.getUserName().equals("默认用户")){
            System.out.println("这个是默认用户");
        }
    }
    // 使用Optional
    Optional.ofNullable(user).filter(u-> StringUtils.equals(u.getUserName(),"默认用户")).ifPresent(
            u-> System.out.println("这个是默认用户")
    );

1.3 map+orElse(提取对象属性值,为null提供默认值)

//optional+orElse
String name = Optional.ofNullable(user).map(User::getUserName).orElse("名字为空");

使用map就可以很好的解决逐层判空的目的,例如本文首图示例,用户对象下有教师对象,教师对象下有年轻教师对象,需要获取年轻教师的名字,常规写法则需要先判断用户是否为null,再逐层判断教师、年轻教师对象是否为null,最终取出想要的值。
例如:

//未使用optional
User user = userService.getById("102BEC97B_4CE3_5E6B_7A4C_B03A26D93A9E");
String name = "";
if (user != null) {
    Teacher teacher = user.getTeacher();
    if (teacher != null) {
        YoungTeahcer youngTeacher = teacher.getYoungTeacher();
        if (youngTeacher != null) {
 	        name = youngTeacher.getName();
        }
    }
}
//使用optional
String youngTeacherName = Optional.ofNullable(user)
    .map(User::getTeacher)
    .map(Teacher::getYoungTeacher)
    .map(YoungTeacher::getName).orElse("未查到");

1.4 map+ ifPresent(可用于逐层判断,执行对象属性值不为null之后的逻辑)

//未用optional
User user = userService.getById("102BEC97B_4CE3_5E6B_7A4C_B03A26D93A9E");
if (user!=null){
    if (StringUtils.isBlank(user.getUserId())){
    //执行用户id不为空的逻辑
        System.err.println("111");
    }
}
//使用optional
Optional.ofNullable(user).map(User::getUserName).ifPresent(t-> System.err.println(t));

1.5 map+orElseThrow(提取对象属性值,为null抛异常)

Optional.ofNullable(user).map(User::getUserName).orElseThrow(() ->  new RuntimeException("查询为空"));

1.6 flatMap(Optional对象进行二次包装)

Optional.ofNullable(byId).flatMap(u -> Optional.ofNullable(u.getUserName()));

2、集合操作(集合判空不建议使用optional)

2.1 集合遍历(建议使用CollectionUtil工具类等,例如new List() 使用optional判断值不是null,则会出现问题。)

List<User> list = userService.list();
//未用optional
if (CollectionUtils.isEmpty(list)){
    list.forEach(u->{
        System.out.println(u.getUserName());
    });
}
//使用optional
Optional.ofNullable(list).orElse(new ArrayList<>()).forEach(user -> System.out.println(user.getUserName()));

四、注意事项

上述例子可以看到Optional的相关api可以结合具体场景组合使用,从而达到判空以及简化代码量的目的,但是Optional在一定场景很好用,但其实很多时候简单的if判断同样可以解决问题,反而使用optional不利于可读性,所以更需要从代码的可读性结合具体场景出发。