內容轉自:http://www.blogjava.net/orangelizq/archive/2007/10/17/153573.htmlhtml
如今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具備相同的屬性值和方法。java
2. Shallow Clone&Deep Clone
Clone是如何完成的呢?Object在對某個對象實施Clone時對其是一無所知的,它僅僅是簡單地執行域對域的copy,這就是Shallow Clone。這樣,問題就來了咯,以Employee爲例,它裏面有一個域hireDay不是基本型別的變量,而是一個reference變量,通過Clone以後就會產生一個新的Date型別的reference,它和原始對象中對應的域指向同一個Date對象,這樣克隆類就和原始類共享了一部分信息,而這樣顯然是不利的,過程下圖所示:
學習
這個時候咱們就須要進行deep Clone了,對那些非基本型別的域進行特殊的處理,例如本例中的hireDay。咱們能夠從新定義Clone方法,對hireDay作特殊處理,以下代碼所示:spa
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的學習筆記。.net
4. Clone()方法的使用
Clone()方法的使用比較簡單,注意以下幾點便可:
a. 何時使用shallow Clone,何時使用deep Clone,這個主要看具體對象的域是什麼性質的,基本型別仍是reference variable
b. 調用Clone()方法的對象所屬的類(Class)必須implements Clonable接口,不然在調用Clone方法的時候會拋出CloneNotSupportedException。code