Java对象拷贝需区分浅拷贝和深拷贝:浅拷贝仅复制一层,引用字段共享;深拷贝则递归创建全新实例。实现上浅拷贝通过Cloneable接口+super.clone(),深拷贝可用序列化、JSON工具或手动clone。选择依据是是否允许共享状态——不可变类型可用浅拷贝,可变集合或需线程安全时须用深拷贝。

Java中对象拷贝不是简单赋值就能完成的,直接用=只是复制引用,两个变量指向同一块堆内存。真正拷贝对象需区分浅拷贝和深拷贝——关键看对象内部的引用类型字段是否也被复制出新实例。
浅拷贝:只复制一层,引用字段仍共享
浅拷贝会创建新对象,基本类型字段值被复制,但引用类型字段(如String、List、自定义对象等)仅复制其引用地址,原对象与拷贝对象仍共用同一个子对象。
实现方式通常为:
- 让类实现
Cloneable接口 - 重写
clone()方法,调用super.clone()
注意:Object.clone()默认就是浅拷贝,无需额外处理引用字段。
立即学习“Java免费学习笔记(深入)”;
深拷贝:完全独立,所有层级都新建
深拷贝要求对象及其内部所有引用类型字段(包括嵌套的嵌套)都生成全新实例,拷贝后两个对象彻底无关,修改互不影响。
常见实现方式有:
-
手动逐层clone:在
clone()中对每个引用字段调用其自身的clone()或构造新对象 - 序列化反序列化:将对象写入字节数组再读出,天然实现深拷贝(要求所有字段所属类都可序列化)
- JSON工具转换:如Gson或Jackson,先转成JSON字符串再解析为新对象(适合POJO,不推荐含复杂逻辑或不可序列化字段的类)
如何判断该用哪种拷贝?
看业务是否允许“共享状态”:
- 如果对象内部全是不可变类型(如String、Integer、LocalDateTime),浅拷贝基本够用
- 如果包含可变集合(ArrayList、HashMap)、自定义实体或缓存字段,且后续可能修改,必须用深拷贝
- 多线程环境下尤其要注意:共享引用可能引发竞态条件,深拷贝能避免意外干扰
一个易错点:String看似引用,实则安全
虽然String是引用类型,但它不可变。浅拷贝时两个对象共用同一个String实例也没问题,因为谁也无法改它。这点常被误认为需要深拷贝,其实不用。
基本上就这些。浅拷贝快但共享风险,深拷贝安全但开销大——选哪个,得看你的对象结构和使用场景。










