使用和避免null
null可能作为默认返回,会导致歧义从而使程序丢失健壮性;
好的方面是,null有时是高效的。
Optional
Optional表示一个可能为 null 的 T 类型引用,它可能包含非null引用(此时为引用存在),也可能什么也不包括(此时为引用缺失),但它从来不会包括null值引用。
Optional possible = Optional.of(5);
possible.isPresent(); // returns true
possible.get(); // returns 5
前置条件:Preconditions
方法调用的前置条件判断更简单。
Preconditions.checkArgument(i>j,"this is error message %s ",333);
相比 Apache Commons 提供的类似方法,我们把 Guava 中的 Preconditions 作为首选。Piotr Jagielski 在
他的博客中简要地列举了一些理由:
• 在静态导入后,Guava 方法非常清楚明晰。checkNotNull 清楚地描述做了什么,会抛出什么异常;
• checkNotNull 直接返回检查的参数,让你可以在构造函数中保持字段的单行赋值风格:this.field = check
NotNull(field)
• 简单的、参数可变的 printf 风格异常信息。鉴于这个优点,在 JDK7 已经引入 Objects.requireNonNull 的
情况下,我们仍然建议你使用 checkNotNull。
在编码时,如果某个值有多重的前置条件,我们建议你把它们放到不同的行,这样有助于在调试时定位。此外,把每个前置条件放到不同的行,也可以帮助你编写清晰和有用的错误消息。
常见Object方法
equals
当一个对象中的字段可以为 null 时,实现 Object.equals 方法会很痛苦,因为不得不分别对它们进行 null 检查(否则直接调用equals方法会报出空异常)。使用 Objects.equal 帮助你执行 null 敏感的 equals 判断,从而避免抛出 NullPointerException。例如:
Objects.equal("a", "a"); // returns true
Objects.equal(null, "a"); // returns false
Objects.equal("a", null); // returns false
Objects.equal(null, null); // returns true
hashCode
用对象的所有字段作散列[hash]运算应当更简单。Guava 的 Objects.hashCode(Object...)会对传入的字段序
列计算出合理的、顺序敏感的散列值。你可以使用 Objects.hashCode(field1, field2, ..., fieldn)来代替手动计算
散列值。
toString
好的 toString 方法在调试时是无价之宝,但是编写 toString 方法有时候却很痛苦。使用 Objects.toStringHelper 可以轻松编写有用的 toString 方法。例如:
Objects.toStringHelper(this).add("x", 1).toString(); // Returns "ClassName{x=1}"
Objects.toStringHelper("MyObject").add("x", 1).toString(); // Returns "MyObject{x=1}"
compare / compareTo -> ComparisonChain
实现一个比较器[Comparator],或者直接实现 Comparable 接口有时也伤不起。考虑一下这种情况:
// 这部分代码太琐碎了,因此很容易搞乱,也很难调试
class Person implements Comparable {
private String lastName;
private String firstName;
private int zipCode;
public int compareTo(Person other) {
int cmp = lastName.compareTo(other.lastName);
if (cmp != 0) {
return cmp;
}
cmp = firstName.compareTo(other.firstName);
if (cmp != 0) {
return cmp;
}
return Integer.compare(zipCode, other.zipCode);
}
}
ComparisonChain 执行一种懒比较:它执行比较操作直至发现非零的结果,在那之后的比较输入将被忽略。
public int compareTo(Foo that) {
return ComparisonChain.start()
.compare(this.aString, that.aString)
.compare(this.anInt, that.anInt)
.compare(this.anEnum, that.anEnum, Ordering.natural().nullsLast())
.result();
}