Java clone() 淺克隆與深度克隆

  現在Clone已經不是一個新鮮詞語了,伴隨着“多莉”的產生這個詞語確實很“火”過一陣子,在Java中也有這麼一個概念,它可以讓我們很方便的“製造”出一個對象的副本來,下面來具體看看java中的Clone機制是如何工作的?
     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具有相同的屬性值和方法。
      2. Shallow Clone&Deep Clone
Clone是如何完成的呢?Object在對某個對象實施Clone時對其是一無所知的,它僅僅是簡單地執行域對域的copy,這就是Shallow Clone。這樣,問題就來了咯,以Employee爲例,它裏面有一個域hireDay不是基本型別的變量,而是一個reference變量,經過Clone之後就會產生一個新的Date型別的reference,它和原始對象中對應的域指向同一個Date對象,這樣克隆類就和原始類共享了一部分信息,而這樣顯然是不利的,過程下圖所示:

 

這個時候我們就需要進行deep Clone了,對那些非基本型別的域進行特殊的處理,例如本例中的hireDay。我們可以重新定義Clone方法,對hireDay做特殊處理,如下代碼所示:

[java] view plain copy
 print?
  1.    class Employee implements Cloneable  
  2.   
  3. {  
  4.         public Object clone() throws CloneNotSupportedException  
  5.         {  
  6.          Employee cloned = (Employee) super.clone();  
  7.       cloned.hireDay = (Date) hireDay.clone()  
  8.       return cloned;  
  9.         }  
  10. }  

 

3. Clone()方法的保護機制

在Object中Clone()是被申明爲protected的,這樣做是有一定的道理的,以Employee

類爲例,通過申明爲protected,就可以保證只有Employee類裏面才能“克隆”Employee對象,原理可以參考我前面關於public、protected、private的學習筆記。

4. Clone()方法的使用

Clone()方法的使用比較簡單,注意如下幾點即可:

a. 什麼時候使用shallow Clone,什麼時候使用deep Clone,這個主要看具體對象的域是什麼性質的,基本型別還是reference variable

b. 調用Clone()方法的對象所屬的類(Class)必須implements Clonable接口,否則在調用Clone方法的時候會拋出CloneNotSupportedException。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章