对象拷贝需创建独立实例而非复制引用。Java中通过clone()实现,浅拷贝仅复制基本类型值和引用地址,共享引用对象;深拷贝递归复制所有层级,完全独立。实现时类需实现Cloneable接口,重写clone()方法并对引用类型字段手动拷贝,或通过序列化实现深拷贝,避免数据污染。

在Java中,对象拷贝是一个常被忽视但非常关键的概念,尤其是在处理对象引用和数据独立性时。理解深拷贝、浅拷贝与克隆机制,有助于避免因共享引用导致的数据污染问题。
什么是对象拷贝?
当我们说“拷贝一个对象”,并不是简单地赋值引用,而是创建一个新对象,其内容与原对象一致。如果只是用=赋值,那只是复制了引用,两个变量指向同一个堆内存中的对象,修改一个会影响另一个。
真正的对象拷贝需要生成独立的对象实例。Java提供了clone()方法作为实现手段,但具体是浅拷贝还是深拷贝,取决于如何实现。
浅拷贝:只复制基本类型,引用类型仍共享
浅拷贝通过调用对象的clone()方法(继承自Object类)实现。它会创建一个新对象,并将原对象的字段逐个复制过去:
立即学习“Java免费学习笔记(深入)”;
- 对于基本数据类型(如int、boolean),直接复制值。
- 对于引用类型(如String、数组、自定义对象),只复制引用地址,不复制被引用的对象。
这意味着,原始对象和拷贝对象中的引用字段指向同一块内存。一旦其中一个修改了引用对象的内容,另一个也会受到影响。
示例:
class Person {String name;
int[] scores;
public Person(String name, int[] scores) {
this.name = name;
this.scores = scores;
}
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
使用时:
int[] arr = {85, 90};Person p1 = new Person("Alice", arr);
Person p2 = (Person) p1.clone();
p2.scores[0] = 100;
System.out.println(p1.scores[0]); // 输出 100,说明共享数组
可见,虽然p2是p1的拷贝,但它们的scores指向同一个数组,这就是浅拷贝的局限。
深拷贝:完全独立,递归复制所有引用对象
深拷贝要求不仅复制对象本身,还要递归复制它所引用的所有对象,直到所有层级都是独立的副本。
要实现深拷贝,不能依赖默认的super.clone(),而要在clone()方法中手动对引用类型字段也进行拷贝。
继续上面的例子:
@Overrideprotected Object clone() throws CloneNotSupportedException {
Person cloned = (Person) super.clone();
cloned.scores = this.scores.clone(); // 对数组进行单独拷贝
return cloned;
}
此时再执行前面的操作,p1.scores[0]就不会被影响,因为两个对象的scores数组是各自独立的副本。
若引用的是复杂对象(比如另一个自定义类),则该类也需要实现深拷贝逻辑,可能涉及序列化或递归调用clone()。
实现克隆的注意事项
Java中实现克隆需遵循一定规范:
- 类必须实现Cloneable接口,否则调用clone()会抛出CloneNotSupportedException。
- Cloneable是一个标记接口,无任何方法,仅表示该类允许克隆。
- 推荐重写clone()为public,方便外部调用。
- 对于包含可变引用字段的类,应明确文档说明是浅拷贝还是深拷贝。
另一种实现深拷贝的方式是通过对象序列化(Serializable),将对象写入字节流再读取,天然实现深度复制,但性能较低且要求所有字段可序列化。
基本上就这些。掌握浅拷贝与深拷贝的区别,能帮助你在设计对象复制逻辑时做出正确选择,避免隐藏的引用共享陷阱。










