万恶的空指针Java8怎么处理的?一文让你减少空指针的出现

2021-12-14 14:07:15 浏览数 (1)

前言

✓ 优质技术好文见专栏 个人公众号,分享一些技术上的文章,以及遇到的坑 当前系列:Java8 新特性 系列 源代码 git 仓库 代码Git 仓库地址

Optional类

​ 到目前为止,臭名昭著的空指针异常是导致Java应用程序失败的最常见原因。 以前,为了解决空指针异常,Google公司著名的Guava项目引入了Optional类, Guava通过使用检查空值的方式来防止代码污染,它鼓励程序员写更干净的代 码。受到Google Guava的启发,Optional类已经成为Java 8类库的一部分。

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

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

创建Optional类对象的方法:

  • Optional.of(T t) : 创建一个 Optional 实例,t必须非空;
  • Optional.empty() : 创建一个空的 Optional 实例
  • Optional.ofNullable(T t):t可以为null

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

  • boolean isPresent() : 判断是否包含对象
  • void ifPresent(Consumer consumer) :如果有值,就执行Consumer 接口的实现代码,并且该值会作为参数传给它。

获取Optional容器的对象:

  • T get(): 如果调用对象包含值,返回该值,否则抛异常
  • T orElse(T other) :如果有值则将其返回,否则返回指定的other对象。
  • T orElseGet(Supplier other) :如果有值则将其返回,否则返回由 Supplier接口实现提供的对象。
  • T orElseThrow(Supplier exceptionSupplier) :如果有值则将其返 回,否则抛出由Supplier接口实现提供的异常。

Optional Coding

代码语言:javascript复制
/**
 * @projectName: Java8
 * @package: OptionalPic
 * @className: OptionalTest
 * @author: 冷环渊 doomwatcher
 * @description: TODO
 * @date: 2021/12/13 0:35
 * @version: 1.0
 */
public class OptionalTest {
    /**
     * @author 冷环渊 Doomwatcher
     * @context:
     * Optional.of(T t) : 创建一个 Optional 实例,t必须非空;
     * Optional.empty() : 创建一个空的 Optional 实例
     * Optional.ofNullable(T t):t可以为null
     * @date: 2021/12/13 0:35
     * @param
     * @return: void
     */
    @Test
    public void test() {
        //创建 Optional
        Girl girl = new Girl();
        Optional<Girl> girl1 = Optional.of(girl);
    }

    /**
     * @author 冷环渊 Doomwatcher
     * @context: Optional.ofNullable(T t):t可以为null
     * @date: 2021/12/13 0:38
     * @param
     * @return: void
     */
    @Test
    public void test2() {
        //创建 Optional
        Girl girl = new Girl();
        girl = null;
        Optional<Girl> girl1 = Optional.ofNullable(girl);
        //如果当前这个 Optional内部封装的t是非空的,则返回内部的t
        //如果内部的t是空的,则返回prElse 方法的参数t1
        Girl girl2 = girl1.orElse(new Girl("赵丽颖"));
        System.out.println(girl2);
    }

    /**
     * @author 冷环渊 Doomwatcher
     * @context: 获取女孩名字
     * @date: 2021/12/13 0:41
     * @param boy
     * @return: java.lang.String
     */
    public String getGirlName(Boy boy) {
        return boy.getGirl().getName();
    }

    /**
     * @author 冷环渊 Doomwatcher
     * @context: 用 Optional优化获取女孩名字
     * @date: 2021/12/13 0:41
     * @param boy
     * @return: java.lang.String
     */
    public String newGetGirlName(Boy boy) {
        Optional<Boy> boyOptional = Optional.ofNullable(boy);
        //此时的 boy1 一定非空
        Boy boy1 = boyOptional.orElse(new Boy(new Girl("乃林")));
        Girl girl = boy1.getGirl();
        Optional<Girl> girl1 = Optional.ofNullable(girl);
        //此时的 girl2 一定非空
        Girl girl2 = girl1.orElse(new Girl("嘉然"));
        return girl2.getName();
    }

    /**
     * @author 冷环渊 Doomwatcher
     * @context: 没有Optional的时候优化获取女孩名字
     * @date: 2021/12/13 0:41
     * @param boy
     * @return: java.lang.String
     */
    public String oldGetGirlName(Boy boy) {
        if (boy != null) {
            Girl girl = boy.getGirl();
            if (girl != null) {
                return girl.getName();
            }
        }
        return null;
    }

    /**
     * @author 冷环渊 Doomwatcher
     * @context:这里调用 应该是空指针因为女孩对象是null
     * @date: 2021/12/13 0:42
     * @param
     * @return: void
     */
    @Test
    public void test3() {
        Boy boy = new Boy();
        String girlName = getGirlName(boy);
        System.out.println(girlName);
    }

    /**
     * @author 冷环渊 Doomwatcher
     * @context:这里调用 这里我们用老的方法进行了处理
     * @date: 2021/12/13 0:42
     * @param
     * @return: void
     */
    @Test
    public void test4() {
        Boy boy = new Boy();
        boy = null;
        String girlName = oldGetGirlName(boy);
        System.out.println(girlName);
    }

    /**
     * @author 冷环渊 Doomwatcher
     * @context:这里调用 这里我们用Optional的方法进行了处理
     * @date: 2021/12/13 0:42
     * @param
     * @return: void
     */
    @Test
    public void test5() {
        // boy 为null 返回乃琳
        // boy的参数 girl为null 返回嘉然
        //如果正常 就返回 正常参数
        Boy boy = new Boy(new Girl("林志玲"));
        //boy = null;
        String girlName = newGetGirlName(boy);
        System.out.println(girlName);
    }

}

总结

到这里我们的 java8新特性就学习完了,这里我们来回顾一下java8 都带给我们了什么帮助

  • Lambda 表达式 新的语法 (函数式接口,方法引用)一切都是为了让我们代码更加的简洁
  • Stream API 我们会发现他和我们的sql很相似,是代码级别的数据处理,同时支持简洁的Lambda
  • Optional 这列类我们一般想不到回去用它,但是很多方法都帮我们用了,我们也可以用它来处理空指针,省去了多个if的嵌套

0 人点赞