深拷贝和浅拷贝
示例:Person 中有两个成员变量,分别是 name 和 age, name 是 String 类型, age 是 int 类 型。代码非常简单,如下所示:
1.public class Person implements Cloneable{ 2.privatint age ; 3. private String name; 4. public Person(int age, String name) { 5. this.age = age; 6. this.name = name; 7. } 8. public Person() {} 9. public int getAge() { 10. return age; 11. } 12. public String getName() { return name; 14. } 15. @Override 16. protected Object clone() throws CloneNotSupportedException { 17. return (Person)super.clone(); 18. } 19.}
由于 age 是基本数据类型,那么对它的拷贝没有什么疑议,直接将一个 4 字节的整数值拷贝过来就行。但是 name 是 String 类型的, 它只是一个引用, 指向一个真正的 String 对象,那么对它的拷贝有两种方式: 直接将原对象中 的 name 的引用值拷贝给新对象的 name 字段, 或者是根据原 Person 对象中的 name 指向的字符串对象创建一个 新的相同的字符串对象,将这个新字符串对象的引用赋给新拷贝的 Person 对象的 name 字段。这两种拷贝方式分别 叫做浅拷贝和深拷贝。深拷贝和浅拷贝的原理如下图所示:

下面通过代码进行验证。如果两个 Person 对象的 name 的地址值相同, 说明两个对象的 name 都指向同一个 String 对象,也就是浅拷贝, 而如果两个对象的 name 的地址值不同, 那么就说明指向不同的 String 对象, 也就 是在拷贝 Person 对象的时候, 同时拷贝了 name 引用的 String 对象, 也就是深拷贝。验证代码如下:
Person p = new Person(23, "zhang"); Person p1 = (Person) p.clone(); String result = p.getName() == p1.getName() ? "clone 是浅拷贝的" : "clone 是深拷贝的"; System.out.println(result);
打印结果:clone 是浅拷贝的
如何进行深拷贝
由上一节的内容可以得出如下结论:如果想要深拷贝一个对象,这个对象必须要实现 Cloneable 接口,实现 clone 方法,并且在 clone 方法内部,把该对象引用的其他对象也要 clone 一份,这就要求这个被引用的对象必须也要实现 Cloneable 接口并且实现 clone 方法。那么,按照上面的结论,实现以下代码 Body 类组合了 Head 类,要想深拷贝 Body 类,必须在 Body 类的 clone 方法中将 Head 类也要拷贝一份。代码如下:
1.static class Body implements Cloneable{ 2. public Head head; 3. public Body() {} 4. public Body(Head head) {this.head = head;} 5. @Override 6. protected Object clone() throws CloneNotSupportedException { 7. Body newBody = (Body) super.clone(); 8. newBody.head = (Head) head.clone(); 9. return newBody; 10. } 11.} 12.static class Head implements Cloneable{ 13. public Face face; 14. public Head() {} 15. @Override 16. protected Object clone() throws CloneNotSupportedException { 17. return super.clone(); 18. } } 19.public static void main(String[] args) throws CloneNotSupportedException { 20. Body body = new Body(new Head(new Face())); 21. Body body1 = (Body) body.clone(); 22. System.out.println("body == body1 : " + (body == body1) ); 23. System.out.println("body.head == body1.head : " + (body.head == body1.head)); 24.} 打印结果为: 1. body == body1 : false 2. body.head == body1.head : fals

浙公网安备 33010602011771号