关于深拷贝和浅拷贝区别,我这里先给结论:
浅拷贝:浅拷贝会在堆上创建一个新的对象(区别于引用拷贝的一点),不过,如果原对象内部的属性是引用类型的话,浅拷贝会直接复制内部对象的引用地址,也就是说拷贝对象混入原对象共用同一个内部对象。
深拷贝:深拷贝会完全复制整个对象,包括这个对象所包含的内部对象。
上面的结论没有完全理解的话也没关系,我们来看一个具体的案例!
浅拷贝
浅拷贝的示例代码如下,我们这里实现Cloneable接口,并重写了clone()方法
clone()方法的实现很简单,直接调用的是父类Object的clone()方法。
public class Address implements Cloneable {
private String name;
// 省略构造函数,Getter&Setter方法
@Override
public Address clone() {
try {
return (Address) super.clone();
}catch(CloneNotSupportedException e){
throw new AssertionError();
}
}
}
public class Person implements Cloneable {
private Address address;
// 省略构造函数,Getter&Setter方法
@Override
public Person clone() {
try{
Person person = (Person) super.clone();
return person;
} catch (CloneNotSupportedException e) {
throw new AssertionError();
}
}
}
测试
Person person1 = new Person(new Address("武汉"));
Person person1Copy = person1.clone();
// true
System.out.println(person1.getAddress() == person1Copy.getAddress());
出输出结构就可以看出,person1的克隆对象和person1使用的仍然是同一个Address对象。
深拷贝
这里我们简单对Person类的clone()方法进行修改,连带着要把Person对象内部的Address对象一起复制。
@Override
public Person clone() {
try {
Person person = (Person) super.clone();
person.setAddress(person.getAddress().clone());
person.setAddress(person.getAddress().clone());
return person;
} catch(CloneNotSupportedException e) {
throw new AssetionError();
}
}
测试:
Person person1 = new Person(new Address("武汉"));
Person person1Copey = person1.clone();
// false
System.out.println(person1.getAddress() == person1Copy.getAddress());
从输出结构就可以看出,显然,person1的克隆对象和person1包含的Address对象已经是不同的了。
那什么是引入拷贝呢?
简单来说,引用拷贝就是两个不同的引用指向同一个对象。
我专门画了一张图来描述浅拷贝,深拷贝,引用拷贝