Java深拷贝和浅拷贝区别 Java对象克隆方法【解析】

浅拷贝仅复制对象本身及基本类型字段,引用类型仍共享;深拷贝递归复制所有层级,完全独立。实现上浅拷贝可重写clone(),深拷贝可用序列化、JSON工具或手动克隆。

浅拷贝只复制对象本身和基本类型字段,引用类型字段仍指向原对象;深拷贝则递归复制所有层级,新对象与原对象完全独立。

浅拷贝:字段值照搬,引用共享

浅拷贝创建一个新对象,把原始对象的字段值逐个复制过去。基本类型(如 int、boolean)会生成新副本;但引用类型(如 List、自定义对象)只是把地址复制过去,新旧对象仍共用同一份堆内存数据。

  • 实现方式常见有:实现 Cloneable 接口并重写 clone() 方法(默认行为就是浅拷贝)
  • 例如:Person

    类含 String name 和 Address addr 字段;浅拷贝后,name 是新字符串(String 不可变,实际效果像深拷贝),但 addr 指向同一个 Address 实例
  • 修改新对象的 addr.city,原对象也会跟着变

深拷贝:彻底隔离,互不影响

深拷贝要求所有嵌套的对象也都要新建实例,确保整个对象图完全独立。没有共享引用,修改任意一方都不会影响另一方。

  • 手动实现:在 clone() 中对每个引用字段调用其 clone() 或 new 创建新对象(需确保所有层级都支持克隆)
  • 序列化方式:将对象写入字节数组再反序列化,天然实现深拷贝(要求所有字段类型都实现 Serializable)
  • JSON 工具(如 Jackson/Gson):先转成 JSON 字符串,再解析成新对象(注意类型信息丢失、不支持非 public 字段或循环引用等限制)

Cloneable 接口不是“魔法”,只是标记

它本身没有方法,也不提供拷贝逻辑。Object.clone() 是 native 方法,仅当类声明实现 Cloneable 时才允许调用,否则抛 CloneNotSupportedException。

  • 必须重写 clone() 并设为 public,否则子类无法调用
  • 浅拷贝是默认行为;想深拷贝,得在重写方法里主动处理引用字段
  • 推荐使用构造函数传参、Builder 模式或不可变设计替代 clone,更清晰可控

选择建议:看场景,别硬套

浅拷贝适合对象结构扁平、引用字段少且不打算修改的情况;深拷贝适合需要完全隔离状态的场景(如缓存快照、多线程安全副本、Undo 操作)。

  • 简单 DTO 对象,字段全是基本类型或 String → 浅拷贝够用
  • 含 List/Map/嵌套 POJO 的复杂对象 → 优先考虑序列化或工具类(如 Apache Commons Lang 的 SerializationUtils)
  • 性能敏感且结构固定 → 手动深拷贝 + 缓存构造器,避免反射开销

基本上就这些。深浅拷贝不是语法特性,而是语义需求;关键在理解“谁变了会影响谁”,再选合适手段实现。