Object類是所有類的始祖,如果一個類沒有使用extends關鍵字明確地指出它的父類,那麼它就會繼承Object類。可以說,所有類都直接或間接地繼承了Object類。本文將對Object類中常用的方法進行介紹。 ...
Object類位於類結構樹的最頂端,所有的類都是它的直接或間接子類,因此所有的類都繼承了Object類的方法,我們可以在需要的時候覆蓋這些方法。下麵是一些將會在本文中討論的Object類的方法:
- protected Object clone() throws CloneNotSupportedException
創建並返回此對象的副本。 - public boolean equals(Object obj)
判斷某個對象是否與這個對象“相等”。 - protected void finalize() throws Throwable
當垃圾回收器將對象從記憶體中清理出去之前要做的清理工作。 - public final Class getClass()
返回對象所屬的類類型。 - public int hashCode()
返回對象的hash值。 - public String toString()
返回對象的字元串表示形式。
下麵的notify,notifyAll和wait方法在同步獨立運行的線程的活動中扮演著不同的角色,本文不會去介紹它們,有關這一部分的內容將會在以後的文章中討論:
- public final void notify()
- public final void notifyAll()
- public final void wait()
- public final void wait(long timeout)
- public final void wait(long timeout, int nanos)
一.equals方法
Object了類中的equals方法用於檢測一個對象是否等於另外一個對象。在Object類中,這個方法將會判斷兩個對象是否具有相同的引用。如果兩個對象具有相同的引用,它們一定是相等的。從這點上看,將其作為預設操作也是合乎情理的。然而,對於大多數類來說,這種判斷並沒有什麼意義。我們在判斷兩個對象是否相等時,應該比較它們的內容,而不僅僅是判斷它們是不是同一個對象。因此,大多數情況下,當我們需要使用equals方法時,都應該對它進行重寫。
為了演示,我們首先編寫一個Apple類:
public class Apple {
private String color;
private int weight;
public void setColor(String color) {
this.color = color;
}
public String getColor() {
return color;
}
public void setWeight(int weight) {
this.weight = weight;
}
public int getWeight() {
return weight;
}
}
當兩個蘋果的重量和顏色一樣時,我們就認為它們是相等的。因此,Apple類的equals方法可以這麼寫:
public boolean equals(Object obj) {
if (obj == null) {
return false;
}
if (this == obj) {
return true;
}
if (this.getClass() != obj.getClass()) {
return false;
}
Apple apple = (Apple) obj;
return weight == apple.getWeight() && Objects.equals(color, apple.getColor());
}
為了防備color為null的情況,上面的例子中使用了Objects.equals(Object a,Object b)方法。如果a和b都是null,這個方法將會返回true;如果其中只有一個參數為null,則返回false;如果a和b都不為null,則會返回a.equals(b)的結果。
實際上,上面的equals還存在一定的問題。不過本文屬於基礎教程系列,因此不會深入講解這其中的問題。有興趣的讀者可以查閱Java規範中對於equals方法的要求以及參考其他深入討論equals方法的文章。
二.hashCode方法
hashCode方法的返回值是根據對象本身所計算出來的散列值(也稱哈希值)。如果兩個對象是相等的,那麼對它們調用hashCode方法得到的返回值也應該是相等的。如果重寫了equals方法,那麼預設的hashCode方法也不再適用。因此,如果重寫了equals方法,則必須同時重寫hashCode方法。在重寫hashCode方法時,原則上只需要保證兩個相等的對象的散列值是相同的即可。不過如何減少衝突以及編寫更高效的哈希函數,可以參考其他文章或查閱電腦演算法書中關於哈希的內容。下麵編寫了一個Apple類的hashCode方法作為示例:
public int hashCode() {
return 7 * (color == null ? 0 : color.hashCode()) + 11 * weight;
}
三.clone方法
如果一個類或它的某個超類實現了Cloneable介面,那麼就可以使用clone()方法從這個類的實例上創建一個副本。在調用clone()方法時,編譯器會檢查這個類是否實現了Cloneable介面。如果沒有,編譯器將會拋出一個CloneNotSupportedException異常。有關異常的內容會在後面的文章中介紹,現在你只需要知道要覆蓋clone()方法,必須將它聲明為:
protected Object clone() throws CloneNotSupportedException
或
public Object clone() throws CloneNotSupportedException
如果調用clone方法的對象實現了Cloneable介面,則繼承自Object類的clone()方法將會創建與原始對象相等的對象,使其具有與原始對象的相應成員變數相同的值。因此,如果想要讓類可以clone,只需要將implements Cloneable添加到類的聲明中即可。
對於某些類,Objects類的clone方法可以正常工作。但是,如果對象包含對外部對象的引用,則可能需要覆蓋clone方法。否則,即使克隆的對象與元對象不是一個對象,但它們內部引用的還是相同的對象。這樣一來,對內部對象所做的更改也會影響到另一個對象。如果需要克隆出一個完全與原對象隔離的新對象,則需要重寫clone方法,將每個內部對象再拷貝一次。
四.finalize方法
finalize方法用於定義在回收對象前要執行的清理工作。Object類的finalize方法什麼也沒做,只有一個空方法體,可以覆蓋finalize方法來定義清理行為,例如釋放資源等。finalize方法不需要也不建議手動調用,它會在垃圾回收器回收對象時自動調用。
五.toString方法
toString方法用於返回表示對象值的字元串。為每個類提供toString方法是一個良好的習慣。
下麵是Object類的toString方法:
public String toString() {
return getClass().getName() + "@" + Integer.toHexString(hashCode());
}
可以看到,Object類的toString方法返回的是類名加對象的hashCode的十六進位表示,中間使用符號@隔開。不過在列印對象的信息時,這個方法的返回值並沒有什麼意義。因此,建議在編寫的每一個類中都覆蓋toString方法。例如為上面的Apple類編寫toString方法:
public String toString() {
return getClass().getName() + "[color = " + color + ",weight = " + weight + "]";
}
六.getClass方法
Class類是一個表示類的信息的類。對對象調用getClass方法會返回一個Class類的實例,用來表示當前對象所屬對象的信息。由於getClass方法是final的,因此無法對它進行重寫。
Class類提供了非常多的方法,例如獲取類名的方法getSimpleName(),獲取父類的方法geuSuperClass(),獲取實現的介面的方法getInterfaces()等。例如,下麵的方法會列印出對象的類名:
void printClassName(Object obj) {
System.out.println("The object's" + " class is " + obj.getClass().getSimpleName());
}
有關Class的內容會在後面有關反射的文章中進行介紹,這裡只需要知道getClass方法的作用即可。