何謂深淺?程度而已。
克?。簭?fù)制一份
被克隆的類實現(xiàn)Cloneable接口,覆蓋clone()方法,返回調(diào)用父類的clone()即可。
public class Person implements Cloneable{
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
//...省略其余代碼
}
然后,克隆時只須調(diào)用我們剛剛覆蓋的clone()。
淺拷貝
現(xiàn)在有Child類:
public class Child{
//...省略其余代碼
}
還有Person類:
public class Person implements Cloneable{
private int age;
private String name;
private Child child;
//...省略其余代碼
}
克隆Person對象p1賦給p2,然后我們看看他們的toString()和hashCode()結(jié)果:
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
結(jié)論:p1,p2成員的值相等,但它們是兩個不同的對象,他們的Child成員是同一個對象。
深拷貝
給Child也實現(xiàn)Cloneable接口并覆蓋clone():
public class Child implements Cloneable{
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();;
}
//...省略其余代碼
}
然后,改變Person類的clone()的實現(xiàn):
public class Person implements Cloneable{
@Override
protected Object clone() throws CloneNotSupportedException {
Person p = (Person)super.clone();
p.setChild((Child)child.clone());
return p;
}
//...省略其余代碼
}
然后再調(diào)用上面的main(String [] args),結(jié)果是:
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
結(jié)論:p1,p2是兩個不同的對象,他們的Child成員也是兩個不同的對象。
總結(jié)
克隆就是復(fù)制一份。
淺拷貝就是簡單復(fù)制值,如果有引用成員,他們的引用成員將是同一個對象,深拷貝則不為然。
為什么呢
可以這樣認(rèn)為,調(diào)用誰的clone()就是就是對誰深拷貝。
在上面淺拷貝的例子中,因為調(diào)用了p1對象的clone(),所以p1和p2是兩個不同的對象(對p1,p2本身深拷貝);在深拷貝的例子中,不但調(diào)用了p1的clone(),也調(diào)用Child類型對象的clone(),所以兩個Child類型對象不同(對所有對象深拷貝)。
思考
基于上述例子,如何實現(xiàn)Person類的clone(),使得只對Child成員進(jìn)行深拷貝,而不對Person本身深拷貝?