文章整理翻译自
文章首发于个人网站:
要说 java 编程中哪个异常是你印象最深刻的,那 nullpointerexception
空指针可以说是臭名昭著的。不要说初级程序员会碰到,
即使是中级,专家级程序员稍不留神,就会掉入这个坑里。
null
引用的发明者 曾在 2009 年作出道歉声明,声明中表示,到目前为止,空指针异常大约给企业已造成数十亿美元的损失。
下面是 tony hoare 的原话:
我将 null 引用的设计称为是一个数十亿美元的错误。1965 那年,我正在用面向对象语言(algol w) 设计首个功能全面的系统。当时我的考量是,确保所有被使用的引用都是安全的,编译器会自动进行检查。但是,我没有抵住诱惑,加入了 null 引用,仅仅是为了实现起来省事。这之后,它导致了数不清的 bug、错误和系统崩溃,也为企业导致了不可估量的损失。
事已至此,我们必须学会面对它。so, 我们要如何防止空指针异常呢?
唯一的办法就是对可能为 null 的对象添加检查。但是 null 检查是繁琐且痛苦的。所以一些比较新的语言为了处理 null 检查,特意添加了特殊的语法,如。
在 或 这样的语言中也被称为
elvis
运算符。
不幸的是,在老版本的 java 中并没有提供这样的语法糖。java8 中在这方面做了改进。所以,这篇文章就特意来介绍一下如何在 java8 中利用新特性来编写防止 nullpointerexception
的发生。
在上篇文章 java8 新特性指导手册 中简单的提了一下如何通过 optional
类来对对象做空校验。接下来,我们再细说一下:
在业务系统中,对象中嵌套对象是经常发生的场景,如下示例代码:
// 最外层对象 class outer { nested nested; nested getnested() { return nested; } } // 第二层对象 class nested { inner inner; inner getinner() { return inner; } } // 最底层对象 class inner { string foo; string getfoo() { return foo; } }
业务中,假设我们需要获取 outer
对象对底层的 inner
中的 foo
属性,我们必须写一堆的非空校验,来防止发生 nullpointerexception
:
// 繁琐的代码 outer outer = new outer(); if (outer != null && outer.nested != null && outer.nested.inner != null) { system.out.println(outer.nested.inner.foo); }
在 java8 中,我们有更优雅的解决方式,那就是使用 optional
是说,我们可以在一行代码中,进行流水式的 map
操作。而 map 方法内部会自动进行空校验:
optional.of(new outer()) .map(outer::getnested) .map(nested::getinner) .map(inner::getfoo .ifpresent(system.out::println); // 如果不为空,最终输出 foo 的值
上面这种方式个人感觉还是有点啰嗦,我们可以利用 suppiler
函数来出一个终极解决方案:
public static <t> optional<t> resolve(supplier<t> resolver) { try { t result = resolver.get(); return optional.ofnullable(result); } catch (nullpointerexception e) { // 可能会抛出空指针异常,直接返回一个空的 optional 对象 return optional.empty(); } }
利用上面的 resolve
方法来重构上述的非空校验代码段:
outer obj = new outer(); // 直接调用 resolve 方法,内部做空指针的处理 resolve(() -> obj.getnested().getinner().getfoo()); .ifpresent(system.out::println); // 如果不为空,最终输出 foo 的值
你需要知道的是,上面这两个解决方案并没传统的 null
检查性能那么高效。但在绝大部分业务场景下,舍弃那么一丢丢的性能来方便编码,是完全可取,
除非是那种对性能有严格要求的场景,我们才不建议使用。
个人觉得,真要拿这点性能说事,还不如去优化优化 sql 语句,业务逻辑等。
如对本文有疑问, 点击进行留言回复!!
【java基础】面试常见问题:类和对象,封装继承多态,final关键字,static关键字,类加载过程,双亲委派模型
荐 Java语言基础之JDK1.8新特性(Lambda表达式、函数式接口、Stream流、新的日期API)
网友评论