• Objects.requireNonNull的意义是什么


    Objects.requireNonNull方法的源码是这样:

    public static  T requireNonNull(T obj) {
        if (obj == null)
            throw new NullPointerException();
        return obj;
    }
    

    曾经我很疑惑,这个判断对象为null再抛出空指针异常的作用是什么,参数为null自然会发生空指针异常,为什么要手动抛出呢?

    它的意义有2点:

    1. 明确代码意图
    2. Fail-fast 快速失败

    明确代码意图

    Objects.requireNonNull让方法的调用者明确某个参数不能为 null。比如,写方法的单元测试,不需要考虑方法参数为null的场景,因为这个场景的是受控的,是不允许的。

    Fail-fast

    (1) Fail-fast 的意思是让代码尽可能早的发生失败,而不是在中途失败。

    Fail-fast 的好处,首先是能立即且稳定的检测出代码的问题,代码立即报错,避免了无用的代码操作;也避免了业务处于一个中间状态,比如业务做到一半出现异常,有可能会出现脏数据。因此,Fail-fast 也提升了程序的稳定性。 在平时写业务代码时,校验逻辑尽可能放在方法前,避免业务在校验不通过前,做了无用操作。比如:

    public String getUserName(User user, String countryCode) {
        queryCountry(countryCode);
        return getUserLastName(user);
    }
    
    public String getUserLastName(User user) {
        return user.getName().getLastName();
    }
    

    user为 null时, queryCountry就是无用操作。

    (2) Fail-fast能快速定位错误位置,也方便Dubug。

    如果user参数为 null,你能定位到哪一行报错,但还不能确保是user还是user.getName的问题,所以你要去调试。如果变量的引用层级更深,定位问题源就更麻烦。如果在源头处控制了user的行为,那么后续操作就令人放心。如果我们在getUserName方法的第一行加上Objects.requireNonNull,这样在报错时我们能准确的捕捉到报错原因,并及时终止业务操作。

    public String getUserName(User user, String countryCode) {
        Objects.requireNonNull(user, "user cannot be null");
        queryCountry(countryCode);
        return getUserLastName(user);
    }
    
    public String getUserLastName(User user) {
        return user.getName().getLastName();
    }
    

    类似用法

    其实在 JDK 中主动对参数判空,然后抛出空指针异常的写法随处可见,它的目的就是提醒方法的调用者,我这个参数不能为空,如果你传入的参数为空了,我能及时终止程序运行,且每次都是相同且稳定的返回结果。
    最后,lombok的@NonNull注解也有这样的功能。比如:

    public int getLength(@NonNull String str) {
        return str.length();
    }
    

    编译后的字节码文件为:

    public int getLength(@NonNull String str) {
        if (str == null) {
            throw new NullPointerException("str is marked non-null but is null");
        } else {
            return str.length();
        }
    }
    
  • 相关阅读:
    线程交替输出(你能想出几种方法)
    【Vue五分钟】五分钟了解组件之间的通信方式
    Linux 64位 C++协程池原理分析及代码实现
    Remove和RemoveLast用法
    深度学习(19):nerf论文公式理解
    netstat 基本使用方法
    环境感知——自动驾驶模型训练(菜鸟版本)
    【241. 为运算表达式设计优先级】
    Docker Buildkit(新增 --mount、--security、--network 等特性)
    三、IPSec
  • 原文地址:https://www.cnblogs.com/cloudrich/p/16923818.html