ullPointerException 是编码过程中必须要处理的防御式检查,我们可能用if(null != user) 或者 Objects.isNull(user)等方式处理,再jdk1.8之后,你可以优雅的处理这个问题
定义
Optional 类是一个可以为null的容器对象。如果值存在则isPresent()方法会返回true,调用get()方法会返回该对象。
Optional 是个容器:它可以保存类型T的值,或者仅仅保存null。Optional提供很多有用的方法,这样我们就不用显式进行空值检测。
Optional 类的引入很好的解决空指针异常。
方法实例演示
例举几个常用的函数,然后进行实际使用分析
of
//创建一个值为张三的String类型的Optional
Optional<String> ofOptional = Optional.of("李四");
//如果我们用of方法创建Optional对象时,所传入的值为null,则抛出NullPointerException
Optional<String> nullOptional = Optional.of(null);
of的用法实际上就是静态工厂方法,示例如下:
@Data
public class Card { private String name; private String number; private Card() {} private Card(String name,String number) { this.name = name; this.number = number; } public static Card of() { return new Card(); } public static Card of(String name,String number) { return new Card(name,number); } }
目前,静态工厂方法比较流行,如果目标类类不需要子类化,非常推荐使用这种方式。
get
如果创建的Optional对象中有值存在则返回此值,如果没有值存在,则会抛出 NoSuchElementException异常
ofNullable
//为指定的值创建Optional对象,不管所传入的值为null不为null,创建的时候都不会报错
Optional<String> nullOptional = Optional.ofNullable(null); Optional<String> noNullOptional = Optional.ofNullable("李四"); System.out.println(nullOptional.get());//抛出异常 NoSuchElementException: No value present System.out.println(noNullOptional.get());//李四
empty
//创建一个空的String类型的Optional对象 Optional<String> emptyOptional = Optional.empty(); System.out.println(emptyOptional .get());//抛出异常 NoSuchElementException
orElse
存在就返回该值,不存在就返回默认值
Optional<String> stringOptional = Optional.of("张三"); System.out.println(stringOptional.orElse("zhangsan"));//张三 Optional<String> emptyOptional = Optional.empty(); System.out.println(emptyOptional.orElse("李四"));//李四
orElseThrow
如果创建的Optional中有值存在,则返回此值,否则抛出一个由指定的Supplier接口生成的异常
Optional<String> stringOptional = Optional.of("张三"); System.out.println(stringOptional.orElseThrow(Exception::new));
map
如果创建的Optional中的值存在,对该值执行提供的Function函数调用
map方法执行传入的lambda表达式参数对Optional实例的值进行修改,修改后的返回值仍然是一个Optional对象
Optional<String> stringOptional = Optional.of("张三"); System.out.println(stringOptional.map(e -> e.toUpperCase()).orElse("不能为空")); stringOptional = Optional.empty(); System.out.println(stringOptional.map(e -> e.toUpperCase()).orElse("不能为空"));
filter
如果创建的Optional中的值满足filter中的条件,则返回包含该值的Optional对象,否则返回一个空的Optional对象
Optional<String> stringOptional = Optional.of("张三"); System.out.println(stringOptional.filter(e -> e.equals("张三")));//Optional[张三] System.out.println(stringOptional.filter(e -> !e.equals("张三")).orElse("李四"));//张三 stringOptional = Optional.empty(); System.out.println(stringOptional.filter(e -> e.equals("张三")).orElse("李四"));//李四
flagMap
flatMap与map(Funtion)方法类似,区别在于flatMap中的mapper返回
值必须是Optional,map方法的mapping函数返回值可以是任何类型T
Optional<String> stringOptional = Optional.of("张三"); System.out.println(stringOptional.flatMap(e -> Optional.of("李四")).orElse("不能为空"));
可能看到这儿之后并没有感觉到多么好用,该写的判断还是要写,咱们继续
实际使用
Person p = new Person("李四",11);//如果 p = null 抛出"年龄不能为空"异常 Integer orElseThrow = Optional.ofNullable(p) .map(s -> s.getAge())//返回参数为年龄的function .map(b ->b + 1)//返回参数为年龄+ 1 的function .filter(m -> m.compareTo(10) == 1)//如果年龄大于10则保留,小于10则过滤掉 .orElseThrow(() -> new Exception("年龄不合法"));//如果为空则抛出该异常 System.out.println(orElseThrow);//12
这种用法可以对单条或多条(配合foreach)能简化大量代码和判断,通过抛出统一异常,使用异常拦截器进行拦截,统一处理,能很大程度上提高开发效率和代码阅读性。