浅拷贝复制对象时,基本类型值独立复制,引用类型仅复制地址引用,导致原对象与副本共享引用对象;深拷贝则递归复制所有引用对象,实现完全独立。选择依据包括对象可变性、嵌套复杂度及性能要求。

在Java中,对象拷贝指的是创建一个已有对象的副本。但根据拷贝方式不同,分为浅拷贝(Shallow Copy)和深拷贝(Deep Copy)。理解它们的区别对避免对象间的数据污染至关重要。
什么是浅拷贝
浅拷贝会创建一个新对象,然后将原对象的字段值逐一复制到新对象中。对于基本数据类型,复制的是实际值;但对于引用类型,复制的只是地址引用,而不是被引用的对象本身。
这意味着,原对象和拷贝对象中的引用字段指向同一个堆内存中的对象。一旦其中一个修改了该共享对象,另一个也会受到影响。
实现浅拷贝的方式通常是让类实现 Cloneable 接口并重写 clone() 方法:
立即学习“Java免费学习笔记(深入)”;
public class Person implements Cloneable {private String name;
private int age;
private Address address; // 引用类型
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
上面代码中,如果调用 clone(),name 和 age 会被独立复制,但 address 字段仍指向原对象的 Address 实例。
什么是深拷贝
深拷贝不仅复制对象本身,还会递归地复制所有引用的对象。这样,原对象和拷贝对象完全独立,互不影响。
继续以上面的 Person 类为例,要实现深拷贝,需要在 clone() 方法中也对 Address 进行拷贝:
@Overrideprotected Object clone() throws CloneNotSupportedException {
Person p = (Person) super.clone();
p.address = (Address) address.clone(); // 假设 Address 也实现了 Cloneable
return p;
}
或者使用其他方式如序列化、JSON 转换等来实现真正的深拷贝。
利用序列化实现深拷贝的思路是:将对象写入字节流再读取出来,生成一个全新的对象实例:
public Person deepCopy() throws IOException, ClassNotFoundException {ByteArrayOutputStream bos = new ByteArrayOutputStream();
ObjectOutputStream oos = new ObjectOutputStream(bos);
oos.writeObject(this);
ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());
ObjectInputStream ois = new ObjectInputStream(bis);
return (Person) ois.readObject();
}
这种方式要求对象及其所有引用对象都实现 Serializable 接口。
如何选择拷贝方式
浅拷贝适用于对象结构简单、引用对象不会被修改,或希望节省内存和提升性能的场景。它速度快,开销小。
深拷贝适用于需要完全隔离两个对象的场景,尤其是当引用对象可能被修改时,能有效防止副作用。
判断使用哪种方式的关键点:
- 是否需要完全独立的对象副本
- 引用对象是否可变(如 ArrayList、自定义类)
- 性能要求是否严格
- 对象结构是否嵌套复杂
基本上就这些。掌握深拷贝与浅拷贝的核心在于理解引用复制与对象复制的区别。合理选择拷贝策略,能有效避免程序中的隐性bug。










