点击上方蓝色“程序猿DD”,选择“设为星标”

回复“资源”获取独家整理的学习资料!

作者 | 李良逸

来源 | blog.imuxuan.com/archives/86

判空灾难

作为搬砖党的一族们,我们对判空一定再熟悉不过了,不要跟我说你很少进行判空,除非你喜欢NullPointerException。

不过NullPointerException对于很多猿们来说,也是Exception家族中最亲近的一员了。

为了避免NullPointerException来找我们,我们经常会进行如下操作。

if (data != null) {    do sth.}

如果一个类中多次使用某个对象,那你可能要一顿操作,so:

“世界第九大奇迹”就这样诞生了。Maybe你会想,项目中肯定不止你一个人会这样一顿操作,然后按下Command+Shift+F,真相就在眼前:

What,我们有接近一万行的代码都是在判空?

好了,接下来,要进入正题了。

NullObject模式

对于项目中无数次的判空,对代码质量整洁度产生了十分之恶劣的影响,对于这种现象,我们称之为“判空灾难”。

那么,这种现象如何治理呢,你可能听说过NullObject模式,不过这不是我们今天的武器,但是还是需要介绍一下NullObject模式。

什么是NullObject模式呢?

In object-oriented computer programming, a null object is an object with no referenced value or with defined neutral ("null") behavior. The null object design pattern describes the uses of such objects and their behavior (or lack thereof).

以上解析来自Wikipedia。

NullObject模式首次发表在“ 程序设计模式语言 ”系列丛书中。一般的,在面向对象语言中,对对象的调用前需要使用判空检查,来判断这些对象是否为空,因为在空引用上无法调用所需方法。

空对象模式的一种典型实现方式如下图所示(图片来自网络):

示例代码如下(命名来自网络,哈哈到底是有多懒):

Nullable是空对象的相关操作接口,用于确定对象是否为空,因为在空对象模式中,对象为空会被包装成一个Object,成为Null Object,该对象会对原有对象的所有方法进行空实现。。

public interfaceNullable{

    booleanisNull();

}

这个接口定义了业务对象的行为。

public interfaceDependencyBaseextendsNullable{

    voidOperation();

}

这是该对象的真实类,实现了业务行为接口DependencyBase与空对象操作接口Nullable。

public classDependencyimplementsDependencyBase, Nullable{

    @Override    publicvoidOperation() {        System.out.print("Test!");    }

    @Override    publicbooleanisNull() {        return false;    }

}

这是空对象,对原有对象的行为进行了空实现。

public classNullObjectimplementsDependencyBase{

    @Override    publicvoidOperation() {        // do nothing    }

    @Override    publicbooleanisNull() {        return true;    }

}

在使用时,可以通过工厂调用方式来进行空对象的调用,也可以通过其他如反射的方式对对象进行调用(一般多耗时几毫秒)在此不进行详细叙述。

public classFactory{

    publicstatic DependencyBase get(Nullable dependencyBase){        if (dependencyBase == null){            return new NullObject();        }        return new Dependency();    }

}

这是一个使用范例,通过这种模式,我们不再需要进行对象的判空操作,而是可以直接使用对象,也不必担心NPE(NullPointerException)的问题。

public classClient{

    publicvoidtest(DependencyBase dependencyBase){        Factory.get(dependencyBase).Operation();    }

}

关于空对象模式,更具体的内容大家也可以多找一找资料,上述只是对NullObject的简单介绍,但是,今天我要推荐的是一款协助判空的插件NR Null Object,让我们来优雅地进行判空,不再进行一顿操作来定义繁琐的空对象接口与空独享实现类。

.NR Null Object

NR Null Object是一款适用于Android Studio、IntelliJ IDEA、PhpStorm、WebStorm、PyCharm、RubyMine、AppCode、CLion、GoLand、DataGrip等IDEA的Intellij插件。其可以根据现有对象,便捷快速生成其空对象模式需要的组成成分,其包含功能如下:

  • 分析所选类可声明为接口的方法;

  • 抽象出公有接口;

  • 创建空对象,自动实现公有接口;

  • 对部分函数进行可为空声明;

  • 可追加函数进行再次生成;

  • 自动的函数命名规范

让我们来看一个使用范例:

怎么样,看起来是不是非常快速便捷,只需要在原有需要进行多次判空的对象中,邮件弹出菜单,选择Generate,并选择NR Null Object即可自动生成相应的空对象组件。

那么如何来获得这款插件呢?

安装方式

可以直接通过IDEA的Preferences中的Plugins仓库进行安装。

选择 Preferences → Plugins → Browse repositories

搜索“NR Null Oject”或者“Null Oject”进行模糊查询,点击右侧的Install,restart IDEA即可。

Optional

还有一种方式是使用Java8特性中的Optional来进行优雅地判空,Optional来自官方的介绍如下:

A container object which may or may not contain a non-null value. If a value is present, isPresent() will return true and get() will return the value.

一个可能包含也可能不包含非null值的容器对象。如果存在值,isPresent()将返回true,get()将返回该值。

话不多说,举个例子。

有如下代码,需要获得Test2中的Info信息,但是参数为Test4,我们要一层层的申请,每一层都获得的对象都可能是空,最后的代码看起来就像这样。

    public String testSimple(Test4 test) {        if (test == null) {            return "";        }        if (test.getTest3() == null) {            return "";        }        if (test.getTest3().getTest2() == null) {            return "";        }        if (test.getTest3().getTest2().getInfo() == null) {            return "";        }        return test.getTest3().getTest2().getInfo();    }

但是使用Optional后,整个就都不一样了。

    public String testOptional(Test test) {        return Optional.ofNullable(test).flatMap(Test::getTest3)                .flatMap(Test3::getTest2)                .map(Test2::getInfo)                .orElse("");    }

1.Optional.ofNullable(test),如果test为空,则返回一个单例空Optional对象,如果非空则返回一个Optional包装对象,Optional将test包装;

    public static <T> Optional<T> ofNullable(T value) {        return value == null ? empty() : of(value);    }

2.flatMap(Test::getTest3)判断test是否为空,如果为空,继续返回第一步中的单例Optional对象,否则调用Test的getTest3方法;

    public<U> Optional<U> flatMap(Function<? super T, Optional<U>> mapper) {        Objects.requireNonNull(mapper);        if (!isPresent())            return empty();        else {            return Objects.requireNonNull(mapper.apply(value));        }    }

3.flatMap(Test3::getTest2)同上调用Test3的getTest2方法;

4.map(Test2::getInfo)同flatMap类似,但是flatMap要求Test3::getTest2返回值为Optional类型,而map不需要,flatMap不会多层包装,map返回会再次包装Optional;

    public<U> Optional<U> map(Function<? super T, ? extends U> mapper) {        Objects.requireNonNull(mapper);        if (!isPresent())            return empty();        else {            return Optional.ofNullable(mapper.apply(value));        }    }

5.orElse("");获得map中的value,不为空则直接返回value,为空则返回传入的参数作为默认值。

public T orElse(T other) {    return value != null ? value : other;}

怎么样,使用Optional后我们的代码是不是瞬间变得非常整洁,或许看到这段代码你会有很多疑问,针对复杂的一长串判空,Optional有它的优势,但是对于简单的判空使用Optional也会增加代码的阅读成本、编码量以及团队新成员的学习成本。毕竟Optional在现在还并没有像RxJava那样流行,它还拥有一定的局限性。

如果直接使用Java8中的Optional,需要保证安卓API级别在24及以上。

你也可以直接引入Google的Guava。(啥是Guava?来自官方的提示)

Guava is a set of core libraries that includes new collection types (such as multimap and multiset), immutable collections, a graph library, functional types, an in-memory cache, and APIs/utilities for concurrency, I/O, hashing, primitives, reflection, string processing, and much more!

引用方式,就像这样:

    dependencies {      compile 'com.google.guava:guava:27.0-jre'      // or, for Android:      api 'com.google.guava:guava:27.0-android'    }

不过IDEA默认会显示黄色,提示让你将Guava表达式迁移到Java Api上。

当然,你也可以通过在Preferences搜索"Guava"来Kill掉这个Yellow的提示。

关于Optional使用还有很多技巧,感兴趣可以查阅Guava和Java8相关书籍和文档。

使用Optional具有如下优点:

  1. 将防御式编程代码完美包装

  2. 链式调用

  3. 有效避免程序代码中的空指针

但是也同样具有一些缺点:

  1. 流行性不是非常理想,团队新成员需要学习成本

  2. 安卓中需要引入Guava,需要团队每个人处理IDEA默认提示,或者忍受黄色提示

  3. 有时候代码阅读看起来可能会如下图所示:

Kotlin

当然,Kotlin以具有优秀的空安全性为一大特色,并可以与Java很好的混合使用,like this:

    test1?.test2?.test3?.test4

如果你已经开始使用了Kotlin,可以不用再写缭乱的防御判空语句。如果你还没有使用Kotlin,并不推荐为了判空优雅而直接转向Kotlin。

本文通过OpenWrite的Markdown转换工具发布

关注我,回复“加群”加入各种主题讨论群

  • DD推荐:编辑器中的翻译神器!参数命名更轻松!

  • 传统网站性能优化的三种手段

  • Spring Boot Admin 2.2.0发布,新增中文展示!

  • IntelliJ IDEA 2019.3发布,2019.2 终成过去式

  • Spring Cloud Hoxton发布 SpringBoot2.2不再孤单

朕已阅 

在 Java 中如何优雅地判空相关推荐

  1. 如何判断map为空_在Java中如何优雅地判空

    来自:李良逸 链接:http://blog.imuxuan.com/archives/86 判空灾难 作为搬砖党的一族们,我们对判空一定再熟悉不过了,不要跟我说你很少进行判空,除非你喜欢 NullPo ...

  2. 如何判断map为空_Java到底如何更优雅地判空

    Java面试笔试面经.Java技术每天学习一点 Java面试 关注不迷路 作者:李良逸 来源:http://blog.imuxuan.com/ 判空灾难 作为搬砖党的一族们,我们对判空一定再熟悉不过了 ...

  3. jeecg中excel导出字段判空处理

    jeecg中excel导出字段判空处理 我们清楚,jeecg 导出 excel 采用的是 easypoi,不知道是否遇到过这种情况: 我们以一个实体属性为例: @Excel(name="问题 ...

  4. java中判断list是否为空的用法(亲测)

    1.如果想判断list是否为空,可以这么判断: if(null == list || list.size() ==0 ){ //为空的情况 }else{ //不为空的情况 } 2.list.isEmp ...

  5. Java集合优雅的判空/非空

    一.乱象 代码中各种同胞写的各种集合判空,很多,很杂乱.大多数是不规范的,而且可能会造成空指针异常. 这篇讲的CollectionUtils工具类是在apache下的, 而不是springframew ...

  6. java如何优雅的判空

    作为搬砖党的一族们,我们对判空一定再熟悉不过了,不要跟我说你很少进行判空,除非你喜欢 NullPointerException. 不过 NullPointerException 对于很多猿们来说,也是 ...

  7. java中如何限制输入非空_项目中的参数限制、非空、以及集合中的验证,你真的会做吗(Java)...

    背景: 在项目管理中,我们通常要对一个个参数做验证, 前端校验,后端加校验,这里为了保证校验的一致性,我们需要将我们的校验写的比较高效率一点,不至于满篇都是is null的进行参数验证,为了成为一个成 ...

  8. 【Java 8】教你用 Java 8中的 Optional 更优雅的判空

    引言 在文章的开头,先说下NPE问题,NPE问题就是,我们在开发中经常碰到的NullPointerException.假设我们有两个类,他们的UML类图如下图所示 图片 在这种情况下,有如下代码 us ...

  9. java判断bean是否为空_总结java中判断对象是否为空的方法

    我们想要判断对象是否为空,像基本类型那样判断是不可以的, ==={} ?这样是错误的,因为这只是比较引用地址是否相同,所以可以采取下面的方法来进行判断. 1.根据for...in遍历对象,如果存在则返 ...

最新文章

  1. FPGA之道(37)Verilog中的编写注意事项
  2. PMCAFF微课堂|诸葛理财联合创始人:互联网金融产品的那些坑
  3. wxWidgets:wxPaintEvent类用法
  4. YUI3下widget的plugin开发
  5. python中的def语句输出1000以内的回文_各种方法测试回文的性能[Python]
  6. Tricks(三十八)—— 在不计算欧式距离的前提下判断点到两点的距离哪个更近
  7. 知识表示学习Trans系列梳理(论文+代码)
  8. 分治法的基本思想_归并排序-分治思想的典型应用
  9. Origin 软件去除demo 水印
  10. 微信小程序性能优化实用建议
  11. saltstack(九)returner
  12. java 阿拉伯数字日期转换为中文大写日期方法_java方法转换大写日期及人民币大写转换方式 .txt...
  13. 如何学习(1):构建全栈式知识结构
  14. 【超分辨率】VDSR论文笔记
  15. 赛元科技EasyCodeCube的使用——Keil环境的配置
  16. 编译原理:语法树,短语,直接短语,句柄
  17. Boost:标记的简单双图bimap的测试程序
  18. burp放包_Burp Suite截断改包发包获取Webshell
  19. 美团的至暗时刻:用户口碑崩塌,食品安全五连击,股价腰斩!
  20. linux 打印两个文件内容相同行和不同行(交集和差集)

热门文章

  1. golang beego orm mysql sqlite3 postgresql 模型字段 数据库类型 对应关系
  2. linux 系统日志 查看被杀掉的进程(占用内存过大)
  3. qemu虚拟机与外部网络的通信
  4. 通过PEB获取模块基址
  5. VS中解决LIBCMTD.lib和uafxcwd.lib冲突(uafxcw.lib LIBCMT.lib冲突)
  6. JDBC在getConnection之前为什么要调用Class.forName
  7. Linux中的中断处理
  8. 大学python选择题题库及答案_大学慕课用Python玩转数据题库及答案
  9. mysql preparedstatement 批量update,PreparedStatement的批量更新的有关问题(只执行最后一条插入语句)...
  10. 稳定wow60级服务器,魔兽怀旧服:永久60级怀旧服是什么体验?