[Google Guava] 1.1-使用和避免null

原文连接 译文连接 译者: 沈义扬 html

   Doug Lea 说,“Null 真糟糕。” java

  当Sir C. A. R. Hoare 使用了null引用后说,”使用它致使了十亿美金的错误。” git

轻率地使用null可能会致使不少使人惊愕的问题。经过学习Google底层代码库,咱们发现95%的集合类不接受null值做为元素。咱们认为, 相比默默地接受null,使用快速失败操做拒绝null值对开发者更有帮助。
编程

此外,Null的含糊语义让人很不舒服。Null不多能够明确地表示某种语义,例如,Map.get(key)返回Null时,可能表示map中的值是null,亦或map中没有key对应的值。Null能够表示失败、成功或几乎任何状况。使用Null之外的特定值,会让你的逻辑描述变得更清晰。 数组

Null确实也有合适和正确的使用场景,如在性能和速度方面Null是廉价的,并且在对象数组中,出现Null也是没法避免的。但相对于底层库来讲,在应用级别的代码中,Null每每是致使混乱,疑难问题和模糊语义的元凶,就如同咱们举过的Map.get(key)的例子。最关键的是,Null自己没有定义它表达的意思。 工具

鉴于这些缘由,不少Guava工具类对Null值都采用快速失败操做,除非工具类自己提供了针对Null值的因变措施。此外,Guava还提供了不少工具类,让你更方便地用特定值替换Null值。 性能

具体案例 学习

不要在Set中使用null,或者把null做为map的键值。使用特殊值表明null会让查找操做的语义更清晰。 google

若是你想把null做为map中某条目的值,更好的办法是 不把这一条目放到map中,而是单独维护一个”值为null的键集合” (null keys)。Map 中对应某个键的值是null,和map中没有对应某个键的值,是很是容易混淆的两种状况。所以,最好把值为null的键分离开,而且仔细想一想,null值的键在你的项目中到底表达了什么语义。 spa

若是你须要在列表中使用null——而且这个列表的数据是稀疏的,使用Map<Integer, E>可能会更高效,而且更准确地符合你的潜在需求。

此外,考虑一下使用天然的null对象——特殊值。举例来讲,为某个enum类型增长特殊的枚举值表示null,好比java.math.RoundingMode就定义了一个枚举值UNNECESSARY,它表示一种不作任何舍入操做的模式,用这种模式作舍入操做会直接抛出异常。

若是你真的须要使用null值,可是null值不能和Guava中的集合实现一块儿工做,你只能选择其余实现。好比,用JDK中的Collections.unmodifiableList替代Guava的ImmutableList

Optional

大多数状况下,开发人员使用null代表的是某种缺失情形:多是已经有一个默认值,或没有值,或找不到值。例如,Map.get返回null就表示找不到给定键对应的值。

Guava用Optional<T>表示可能为null的T类型引用。一个Optional实例可能包含非null的引用(咱们称之为引用存在),也可能什么也不包括(称之为引用缺失)。它从不说包含的是null值,而是用存在或缺失来表示。但Optional从不会包含null值引用。

1 Optional<Integer> possible = Optional.of(5);
2  
3 possible.isPresent(); // returns true
4  
5 possible.get(); // returns 5

Optional无心直接模拟其余编程环境中的”可选” or “可能”语义,但它们的确有类似之处。

Optional最经常使用的一些操做被罗列以下:

建立Optional实例(如下都是静态方法):

Optional.of(T) 建立指定引用的Optional实例,若引用为null则快速失败
Optional.absent() 建立引用缺失的Optional实例
Optional.fromNullable(T) 建立指定引用的Optional实例,若引用为null则表示缺失

用Optional实例查询引用(如下都是非静态方法):

boolean isPresent() 若是Optional包含非null的引用(引用存在),返回true
T get() 返回Optional所包含的引用,若引用缺失,则抛出java.lang.IllegalStateException
T or(T) 返回Optional所包含的引用,若引用缺失,返回指定的值
T orNull() 返回Optional所包含的引用,若引用缺失,返回null
Set<T> asSet() 返回Optional所包含引用的单例不可变集,若是引用存在,返回一个只有单一元素的集合,若是引用缺失,返回一个空集合。

使用Optional的意义在哪儿?

使用Optional除了赋予null语义,增长了可读性,最大的优势在于它是一种傻瓜式的防御。Optional迫使你积极思考引用缺失的状况,由于你必须显式地从Optional获取引用。直接使用null很容易让人忘掉某些情形,尽管FindBugs能够帮助查找null相关的问题,可是咱们仍是认为它并不能准确地定位问题根源。

如同输入参数,方法的返回值也多是null。和其余人同样,你绝对极可能会忘记别人写的方法method(a,b)会返回一个null,就好像当你实现method(a,b)时,也极可能忘记输入参数a能够为null。将方法的返回类型指定为Optional,也能够迫使调用者思考返回的引用缺失的情形。

其余处理null的便利方法

当你须要用一个默认值来替换可能的null,请使用Objects.firstNonNull(T, T) 方法。若是两个值都是null,该方法会抛出NullPointerException。Optional也是一个比较好的替代方案,例如:Optional.of(first).or(second).

还有其它一些方法专门处理null或空字符串:emptyToNull(String)nullToEmpty(String)isNullOrEmpty(String)。咱们想要强调的是,这些方法主要用来与混淆null/空的API进行交互。当每次你写下混淆null/空的代码时,Guava团队都泪流满面。(好的作法是积极地把null和空区分开,以表示不一样的含义,在代码中把null和空同等对待是一种使人不安的坏味道。

相关文章
相关标签/搜索