浅拷贝复制对象时仅复制基本类型值和引用地址,原对象与副本共享引用对象,修改会影响彼此;深拷贝递归复制所有层级,完全独立。实现上,浅拷贝可通过实现Cloneable接口重写clone()方法,深拷贝需手动克隆引用对象或使用序列化、第三方库。选择依据是对象结构复杂度及是否需完全隔离,深拷贝性能开销更大。

在Java中,理解对象的深拷贝和浅拷贝关键在于搞清楚对象引用和对象实例之间的关系。当你复制一个对象时,你可能只想复制它的基本类型字段,也可能需要连同它内部引用的对象一起复制。这就引出了浅拷贝和深拷贝的区别。
什么是浅拷贝
浅拷贝是指创建一个新对象,这个新对象会拥有原对象中各个字段的值。对于基本数据类型,复制的是实际的值;而对于引用类型,复制的只是引用地址,也就是说,原对象和拷贝对象中的引用字段指向的是同一个对象。
这意味着:如果通过拷贝对象修改了某个引用类型的属性,原始对象也会受到影响,因为它们共享同一个引用对象。
实现方式通常可以通过重写clone()方法并实现Cloneable接口来完成:
立即学习“Java免费学习笔记(深入)”;
- 类实现Cloneable接口
- 重写clone()方法,调用super.clone()
注意:super.clone()默认执行的就是浅拷贝。
什么是深拷贝
深拷贝不仅复制对象本身,还会递归地复制对象中所有引用类型的成员变量,确保原对象和拷贝对象完全独立。即使原对象中的字段是对象引用,拷贝后的新对象也会拥有这些引用对象的“副本”,而不是共享。
这样一来,修改拷贝对象不会影响原始对象,反之亦然。
实现深拷贝有几种常见方式:
- 手动重写 clone() 方法:在克隆时,对每个引用类型的字段也调用其 clone() 方法(前提是这些类也支持深拷贝)
- 序列化与反序列化:将对象序列化成字节流再反序列化回来,生成一个全新的对象树,这是实现深拷贝最彻底的方式之一
- 使用第三方库:如 Gson、Jackson(适用于POJO转JSON再转对象)、Apache Commons Lang 的 SerializationUtils 等
举个例子说明区别
假设有一个Person类,包含姓名(String)和住址(Address对象):
- 做浅拷贝时,新的 Person 对象的 name 是独立的(String不可变),但 address 引用和原对象相同
- 做深拷贝时,address 也会被复制一份,两个 Person 指向不同的 Address 实例
如果你改了拷贝后的 Person 的住址信息,浅拷贝会影响原对象,深拷贝则不会。
如何选择使用哪种拷贝
根据实际需求决定:
- 如果对象结构简单,且引用的对象不会被修改,浅拷贝足够且效率高
- 如果需要完全隔离原对象和副本,尤其是涉及可变对象(如List、自定义类等),应使用深拷贝
- 注意性能开销:深拷贝比浅拷贝更耗资源,特别是对象层级深或数据量大时
基本上就这些。关键是明白:浅拷贝“复制一层”,深拷贝“复制全部”。理解这一点,就能在实际开发中合理选择复制策略。不复杂但容易忽略细节。










