在Java中,深拷贝和浅拷贝是对象复制的两种方式,主要区别在于对对象内部的引用类型的处理上。
浅拷贝是指创建一个新的对象,但这个新对象的属性(包括引用类型的属性)仍然指向原来对象的属性。换言之,如果原对象中的属性是一个引用类型,那么浅拷贝只会复制这个引用的地址,新旧对象会共享同一块内存区域。因此,修改其中一个对象的引用类型属性时,另一个对象的相同属性也会受到影响。优点在于省资源。适合的场景是只读取不写入的情况,对象结构简单或希望节省资源。示例代码如下所示:
class Person implements Cloneable {
String name;
Address address; // 假设Address是一个类
public Person clone() {
try {
return (Person) super.clone();
} catch (CloneNotSupportedException e) {
throw new AssertionError(); // 无法克隆的情况不应该发生
}
}
}
对于每个引用类型的属性,手动调用其拷贝方法(如果该类也支持深拷贝)来创建新的实例。
深拷贝不仅会创建一个新对象,还会递归地为所有引用类型的属性创建副本,确保源对象和拷贝对象之间完全独立,修改一个对象不会影响到另一个对象。适合的场景是需要写入操作的情况,需要完全独立的副本,避免修改时的相互影响。实现深拷贝的方法有多种,可以重写clone()方法、使用第三方库、手动实现深拷贝逻辑以及序列化方式。
要分辨代码中的深浅拷贝,可以通过赋值和方法参数传递来判断。在Java中,方法参数传递本质上是值传递。当对象作为参数传递给方法时,传递的是对象引用的副本(即指针的拷贝),而不是对象本身的新拷贝。因此,方法内对对象内容的修改会影响原始对象,这是浅拷贝行为。但如果方法内部创建了对象的新实例并返回,则可以实现深拷贝。
注意:基本数据类型与引用类型有区别,基本数据类型的赋值总是按值传递,不会涉及深浅拷贝问题。