Java常用類(一)之Object類詳解


大家都知道Object是所有類的父類,任何類都默認繼承Object

理論上Object類是所有類的父類,即直接或間接的繼承java.lang.Object類。由於所有的類都繼承在Object類,因此省略了extends Object關鍵字。
該類中主要有以下方法: toString(),getClass(),equals(),clone(),finalize(), 其中toString(),getClass(),equals是其中最重要的方法。

注意:

    Object類中的getClass(),notify(),notifyAll(),wait()等方法被定義為final類型,因此不能重寫。

一、clone()方法

保護方法,實現對象的淺復制,只有實現了Cloneable接口才可以調用該方法,否則拋出CloneNotSupportedException異常。

主要是JAVA里除了8種基本類型傳參數是值傳遞,其他的類對象傳參數都是引用傳遞,我們有時候不希望在方法里講參數改變,這是就需要在類中復寫clone方法(實現深復制)。

 創建並返回此對象的一個副本。“副本”的准確含義可能依賴於對象的類。

1.1、clone與copy的區別

    假設現在有一個Employee對象,Employee tobby =new Employee(“CMTobby”,5000)
    通常我們會有這樣的賦值Employee cindyelf=tobby,這個時候只是簡單了copy了一下reference,cindyelf和tobby都指向內存中同一個object,
    這樣cindyelf或者tobby的一個操作都可能影響到對方。打個比方,如果我們通過cindyelf.raiseSalary()方法改變了salary域的值,那么tobby通過
    getSalary()方法得到的就是修改之后的salary域的值,顯然這不是我們願意看到的。我們希望得到tobby的一個精確拷貝,同時兩者互不影響,這時候
    我們就可以使用Clone來滿足我們的需求。Employee cindy=tobby.clone(),這時會生成一個新的Employee對象,並且和tobby具有相同的屬性值和方法。

1.2、Shallow Clone與Deep Clone

    Clone是如何完成的呢?Object在對某個對象實施Clone時對其是一無所知的,它僅僅是簡單地執行域對域的copy,這就是Shallow Clone。這樣,問題就來了咯。
    以Employee為例,它里面有一個域hireDay不是基本數據類型的變量,而是一個reference變量,經過Clone之后就會產生一個新的Date型的reference,
    它和原始對象中對應的域指向同一個Date對象,這樣克隆類就和原始類共享了一部分信息,而這樣顯然是不利的,過程下圖所示:
   
    這個時候我們就需要進行deep Clone了,對那些非基本型別的域進行特殊的處理,例如本例中的hireDay。我們可以重新定義Clone方法,對hireDay做特殊處理,如下代碼所示:
<span data-wiz-span="data-wiz-span" style="font-size: 1.167rem;">   class Employee implements Cloneable  {  
        public Object clone() throws CloneNotSupportedException  {  
         Employee cloned = (Employee) super.clone();  
      cloned.hireDay = (Date) hireDay.clone()  
      return cloned;  
        }  
   }</span>

1.3、clone方法的保護機制

    在Object中Clone()是被聲明為protected的,這樣做是有一定的道理的,以Employee。
    類為例,通過聲明為protected,就可以保證只有Employee類里面才能“克隆”Employee對象

1.4、clone方法的使用

    Clone()方法的使用比較簡單,注意如下幾點即可:
 
        什么時候使用shallow Clone,什么時候使用deep Clone,這個主要看具體對象的域是什么性質的,基本型別還是reference variable
        調用Clone()方法的對象所屬的類(Class)必須implements Clonable接口,否則在調用Clone方法的時候會拋出CloneNotSupportedException
更加詳細的解釋:點擊查看

二、toString()方法

Object 類的 toString 方法返回一個字符串,該字符串由類名(對象是該類的一個實例)、at 標記符“@”和此對象哈希碼的無符號十六進制表示組成
該方法用得比較多,一般子類都有覆蓋。

    例子:
<span data-wiz-span="data-wiz-span" style="font-size: 1.167rem;">package com.cal.toString;  
  
public class Test1 {  
    public static void main(String[] args){  
        Object o1 = new Object();  
        System.out.println(o1.toString());  
    }  
}  </span>

 

        結果:java.lang.Object@7852e922

三、getClass()方法

 返回次Object的運行時類類型。
不可重寫,要調用的話,一般和getName()聯合使用,如getClass().getName();

四、finalize()方法


該方法用於釋放資源。因為無法確定該方法什么時候被調用,很少使用。

Java允許在類中定義一個名為finalize()的方法。它的工作原理是:一旦垃圾回收器准備好釋放對象占用的存儲空間,將首先調用其finalize()方法。並且在下一次垃圾回收動作發生時,才會真正回收對象占用的內存。

關於垃圾回收,有三點需要記住:

  1、對象可能不被垃圾回收。只要程序沒有瀕臨存儲空間用完的那一刻,對象占用的空間就總也得不到釋放。

  2、垃圾回收並不等於“析構”。

  3、垃圾回收只與內存有關。使用垃圾回收的唯一原因是為了回收程序不再使用的內存。

finalize()的用途:

  無論對象是如何創建的,垃圾回收器都會負責釋放對象占據的所有內存。這就將對finalize()的需求限制到一種特殊情況,即通過某種創建對象方式以外的方式為對象分配了存儲空間。

  不過這種情況一般發生在使用“本地方法”的情況下,本地方法是一種在Java中調用非Java代碼的方式。

五、equals()方法

Object中的equals方法是直接判斷this和obj本身的值是否相等,即用來判斷調用equals的對象和形參obj所引用的對象是否是同一對象,
所謂同一對象就是指內存中同一塊存儲單元,如果this和obj指向的hi同一塊內存對象,則返回true,如果this和obj指向的不是同一塊內存,則返回false。
注意:即便是內容完全相等的兩塊不同的內存對象,也返回false。

          如果是同一塊內存,則object中的equals方法返回true,如果是不同的內存,則返回false

          如果希望不同內存但相同內容的兩個對象equals時返回true,則我們需要重寫父類的equal方法

          String類已經重寫了object中的equals方法(這樣就是比較內容是否相等了)

六、hashCode()方法

返回該對象的哈希碼值

該方法用於哈希查找,可以減少在查找中使用equals的次數,重寫了equals方法一般都要重寫hashCode方法。這個方法在一些具有哈希功能的Collection中用到。

一般必須滿足obj1.equals(obj2)==true。可以推出obj1.hash- Code()==obj2.hashCode(),但是hashCode相等不一定就滿足equals。不過為了提高效率,應該盡量使上面兩個條件接近等價。

如果不重寫hashcode(),在HashSet中添加兩個equals的對象,會將兩個對象都加入進去。

七、wait()方法

1)wait()

  

2)wait(long timeout)

  

3)wait(long timeout,int naos)

  

  什么意思呢?

    

  方法中的異常:

       

wait方法就是使當前線程等待該對象的鎖,當前線程必須是該對象的擁有者,也就是具有該對象的鎖。wait()方法一直等待,直到獲得鎖或者被中斷。wait(long timeout)設定一個超時間隔,

如果在規定時間內沒有獲得鎖就返回。

 

調用該方法后當前線程進入睡眠狀態,直到以下事件發生。

(1)其他線程調用了該對象的notify方法。

(2)其他線程調用了該對象的notifyAll方法。

(3)其他線程調用了interrupt中斷該線程。

(4)時間間隔到了。

此時該線程就可以被調度了,如果是被中斷的話就拋出一個InterruptedException異常。

八、notify()方法

該方法喚醒在該對象上等待的某個線程。

九、notifyAll方法

該方法喚醒在該對象上等待的所有線程。

 

喜歡就點推薦哦!

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM