淺談Java克隆
何謂深淺?程度而已。
克隆:複製一份
被克隆的類實現Cloneable
介面,覆蓋clone()
方法,返回呼叫父類的clone()
即可。
public class Person implements Cloneable{ @Override protected Object clone() throws CloneNotSupportedException { return super.clone(); } //...省略其餘程式碼 }
然後,克隆時只須呼叫我們剛剛覆蓋的clone()
。
淺拷貝
現在有Child
類:
public class Child{ //...省略其餘程式碼 }
還有Person
類:
public class Person implements Cloneable{ private int age; private String name; private Child child; //...省略其餘程式碼 }
克隆Person物件p1賦給p2,然後我們看看他們的toString()
和hashCode()
結果:
public class Main { public static void main(String[] args) throws CloneNotSupportedException { Person p1 = new Person("zhangsan",23); p1.setChild(new Child("xiaoming", 1);); //克隆操作 Person p2 = p1.clone(); //列印看看 System.out.println(p1+", "+p1.hashCode()+", "+p1.getChild().hashCode()); System.out.println(p2+", "+p2.hashCode()+", "+p2.getChild().hashCode()); } }
輸出:
Person [age=23, name=zhangsan, value=666, child=Child [name=xiaoming, age=1]], 118352462, 1550089733 Person [age=23, name=zhangsan, value=666, child=Child [name=xiaoming, age=2]], 865113938, 1550089733
結論:p1,p2成員的值相等,但它們是兩個不同的物件,他們的Child成員是同一個物件。
深拷貝
給Child
也實現Cloneable
介面並覆蓋clone()
:
public class Child implements Cloneable{ @Override protected Object clone() throws CloneNotSupportedException { return super.clone();; } //...省略其餘程式碼 }
然後,改變Person
類的clone()
的實現:
public class Person implements Cloneable{ @Override protected Object clone() throws CloneNotSupportedException { Person p = (Person)super.clone(); p.setChild((Child)child.clone()); return p; } //...省略其餘程式碼 }
然後再呼叫上面的main(String [] args)
,結果是:
Person [age=23, name=zhangsan, value=666, child=Child [name=xiaoming, age=1]], 118352462, 1550089733 Person [age=23, name=zhangsan, value=666, child=Child [name=xiaoming, age=2]], 865113938, 1442407170
結論:p1,p2是兩個不同的物件,他們的Child成員也是兩個不同的物件。
總結
克隆就是複製一份。
淺拷貝就是簡單複製值,如果有引用成員,他們的引用成員將是同一個物件,深拷貝則不為然。
為什麼呢
可以這樣認為,呼叫誰的clone()
就是就是對誰深拷貝。
在上面淺拷貝的例子中,因為呼叫了p1
物件的clone()
,所以p1
和p2
是兩個不同的物件(對p1,p2本身深拷貝);在深拷貝的例子中,不但呼叫了p1
的clone()
,也呼叫Child
型別物件的clone()
,所以兩個Child
型別物件不同(對所有物件深拷貝)。
思考
基於上述例子,如何實現Person
類的clone()
,使得只對Child
成員進行深拷貝,而不對Person
本身深拷貝?