专栏名称: 码小辫
给程序员和编程爱好者分享计算机编程电子书以及相关的学习资源
目录
相关文章推荐
创伙伴  ·  机器人养老,离普通人有多远? ·  16 小时前  
连州点点网  ·  重大利好!广东重磅发布→ ·  昨天  
连州点点网  ·  重大利好!广东重磅发布→ ·  昨天  
独角兽智库  ·  机器人关注方向 ·  3 天前  
51好读  ›  专栏  ›  码小辫

改变习惯性 !=null 的判断,只需一秒!

码小辫  · 公众号  ·  · 2025-02-07 17:10

正文

空指针异常是出现频率比较高的bug,在出现空指针时,很多小伙伴都是习惯性地加一个 !=null 的判断,这个bug就解决了。

当代码中频繁出现 !=null 的判断时,我们就会很头疼,能不能高效、优雅地做这个判断?

答案当然是可以的。

第一步

当我们要做 !=null 的判断时,请停顿一下,看一下我们要做判断的这个数据是什么类型?

数据类型无非就是String字符串、Object/自定义对象、List集合、Array数组、Map集合等类型。

第二步

思考这个数据类型对应的工具类有哪些?

String类型,对应的工具类有StringUtils。

Object对象,对应的工具类有ObjectUtils。

Array数组,对应的工具类有Arrays。

List集合、Map集合对应的工具类有Collections、CollectionUtils等。

这些工具类都是Java、Spring框架自带的工具类。

第三步

使用对应类型的工具类做判断。

1、如果这个数据是String类型时,请使用StringUtils工具类。

String str = "";
StringUtils.isEmpty(str);  // true

StringUtils工具类比较有针对性,是针对String字符串的工具类。

public static boolean isEmpty(@Nullable Object str) {
   return str == null || "".equals(str);
}

在isEmpty方法中,既有为null的判断,也有是否等于空字符串的判断。

2、如果这个数据是Object类型,请使用ObjectUtils工具类。

Object obj = null;
ObjectUtils.isEmpty(obj); // true

3、如果这个数据是Map类型,也可以使用ObjectUtils工具类。

Map map = Collections.emptyMap();
ObjectUtils.isEmpty(map);// true

4、如果这个数据是List类型,还可以使用ObjectUtils工具类。

List list =Collections.EMPTY_LIST;
ObjectUtils.isEmpty(list); // true

5、如果这个数据是数组类型,依旧可以使用ObjectUtils工具类。

// 数组
Object[] objArr = null;
ObjectUtils.isEmpty(objArr); // true

ObjectUtils 中的isEmpty()这一个方法,分别可以对字符串、数组、Map集合、List集合进行是否等于null的判断。

这个isEmpty方法为什么能判断这么多种数据类型呢?一起看下它的源码。

public static boolean isEmpty(@Nullable Object obj) {
    // 判断obj是否为null,如果是直接f
    if (obj == null) {
      return true;
    }
    // 判断obj是否是Optional的子类
    if (obj instanceof Optional) {
      // 如果是,则调用isPresent方法判断是否为null
      return !((Optional) obj).isPresent();
    }
    // 判断obj是否是CharSequence的子类
    if (obj instanceof CharSequence) {
      // 如果是,则获取长度,长度等于0时,就认为这个obj是空字符串
      return ((CharSequence) obj).length() == 0;
    }
    // 判断obj是否为数组
    if (obj.getClass().isArray()) {
      // 数组的长度等于0就认为这个数组是空数组
      return Array.getLength(obj) == 0;
    }
    // 判断obj是否为Collection集合的子类
    if (obj instanceof Collection) {
      // 用Collection子类的isEmpty方法判断集合是否为空
      return ((Collection) obj).isEmpty();
    }
    // 判断obj是否为Map接口的子类
    if (obj instanceof Map) {
      // 如果是,则进行强转,并用子类的isEmpty方法判断集合是否为空
      return ((Map) obj).isEmpty();
    }

    // else
    return false;
}

在这个静态方法中,首先对传入的obj对象进行是否等于null的判断,如果是则返回。如果不是null,对obj进行数据类型进行定位,然后根据数据类型进行判断。

在这个方法中,封装了Optional、CharSequence、Array、Collection、Map数据类型,几乎涵盖所有的数据类型。

通过这段源码,我们也可以看出,它对复杂类型的集合的判断存在一些缺陷。也就是说它只判断了集合的长度,集合的长度为0,就认为集合是空的。

// 创建一个只有一个元素的List集合
List list = Collections.singletonList(null);
ObjectUtils.isEmpty(list); // false

上面的这段代码,我们创建了一个只有一个元素、且这个元素为null的List集合。

如果是对象数组,数组里有一个对象,但这一个对象为null,这个判断就失灵了。

ObjectUtils 类中对对象数组的判断是另外一个isEmpty方法。

public static boolean isEmpty(@Nullable Object[] array) {
return array == null || array.length == 0;
}

因此在这2种情况下,我们再使用ObjectUtils的isEmpty方法就不合适了,我们需要对集合或数组里的每一个元素进行判断是否为null。







请到「今天看啥」查看全文