深拷贝可避免Java对象拷贝时引用共享问题,浅拷贝仅复制基本类型,引用类型仍共用内存地址;深拷贝需递归复制所有层级对象,可通过拷贝构造函数、序列化或第三方库如SerializationUtils实现,确保副本独立。
Java对象拷贝时,如果处理不当,很容易导致原始对象和副本共享引用类型字段,造成一方修改影响另一方的问题。要避免这种情况,关键在于正确实现深拷贝。
浅拷贝只复制对象的基本数据类型字段,而引用类型的字段仍指向原对象的内存地址。这意味着两个对象会共享同一个引用对象,修改其中一个会影响另一个。
深拷贝则会递归复制所有层级的对象,包括引用类型字段,确保副本完全独立。
例如:
假设一个Person类包含String name和Address address。浅拷贝后,两个Person实例的address字段指向同一个Address对象;深拷贝则会创建新的Address实例。最直接的方式是通过构造函数或静态工厂方法,显式复制每个字段,对引用类型调用其拷贝逻辑。
示例:
public class Person {这种方式控制力强,易于调试,适合字段不多或结构固定的类。
将对象序列化为字节流再反序列化,可以得到一个全新的对象树,天然避免引用共享。
前提是类必须实现Serializable接口,且所有引用的成员也支持序列化。
示例代码:
public static这种方法适用于复杂对象图,但性能较低,且要求可序列化。
Apache Commons Lang提供了SerializationUtils,封装了基于序列化的深拷贝。
使用方式简单:
Person copy = SerializationUtils.clone(person);此外,像Kryo这样的高性能序列化库也可用于快速深拷贝,尤其适合频繁拷贝场景。
基本上就这些。选择哪种方式取决于性能要求、对象结构和是否支持序列化。核心原则是:确保引用类型字段也被独立复制,才能真正避免共享问题。