當(dāng)一個對象中的字段可以為 null 時,實(shí)現(xiàn) Object.equals 方法會很痛苦,因?yàn)椴坏貌环謩e對它們進(jìn)行 null 檢查。使用 Objects.equal 幫助你執(zhí)行 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
注意:JDK7 引入的 Objects 類提供了一樣的方法 Objects.equals。
用對象的所有字段作散列[hash]運(yùn)算應(yīng)當(dāng)更簡單。Guava 的 Objects.hashCode(Object...)會對傳入的字段序列計算出合理的、順序敏感的散列值。你可以使用 Objects.hashCode(field1, field2, …, fieldn)來代替手動計算散列值。
注意:JDK7 引入的 Objects 類提供了一樣的方法 Objects.hash(Object...)
好的 toString 方法在調(diào)試時是無價之寶,但是編寫 toString 方法有時候卻很痛苦。使用 Objects.toStringHelper 可以輕松編寫有用的 toString 方法。例如:
// Returns "ClassName{x=1}"
Objects.toStringHelper(this).add("x", 1).toString();
// Returns "MyObject{x=1}"
Objects.toStringHelper("MyObject").add("x", 1).toString();
實(shí)現(xiàn)一個比較器[Comparator],或者直接實(shí)現(xiàn) Comparable 接口有時也傷不起。考慮一下這種情況:
class Person implements Comparable<Person> {
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);
}
}
這部分代碼太瑣碎了,因此很容易搞亂,也很難調(diào)試。我們應(yīng)該能把這種代碼變得更優(yōu)雅,為此,Guava 提供了 ComparisonChain。
ComparisonChain 執(zhí)行一種懶比較:它執(zhí)行比較操作直至發(fā)現(xiàn)非零的結(jié)果,在那之后的比較輸入將被忽略。
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();
}
這種 Fluent 接口風(fēng)格的可讀性更高,發(fā)生錯誤編碼的幾率更小,并且能避免做不必要的工作。更多 Guava 排序器工具可以在下一節(jié)里找到。