浅拷贝复制对象及其基本类型字段值,引用类型仅复制地址;深拷贝递归复制所有引用对象,形成独立副本。1. 浅拷贝通过clone()等方法实现,引用字段指向同一内存空间,修改相互影响;2. 深拷贝需手动逐层克隆、序列化或使用第三方库,确保嵌套对象独立;3. 区别在于引用类型处理方式不同,浅拷贝共享数据,深拷贝完全隔离;4. 选择策略:结构简单用手动克隆,复杂嵌套则推荐序列化或工具库。理解两者差异有助于避免数据污染和并发问题。

在Java中,对象拷贝分为深拷贝和浅拷贝,这两个概念看起来差不多,但实际使用时差别挺大。简单来说,浅拷贝只复制对象本身及其基本数据类型的字段值,而引用类型字段只是复制了引用地址;而深拷贝会递归复制对象内部的所有引用对象,形成一个完全独立的副本。

如果你不注意这个区别,可能会在修改对象属性时遇到意想不到的问题。
浅拷贝是指创建一个新对象,然后将原对象中的所有字段复制到新对象中。如果字段是基本类型(如int、boolean等),直接复制其值;如果是引用类型,则只复制引用地址,两个对象指向的是同一个内存空间。
立即学习“Java免费学习笔记(深入)”;

比如:
class Person implements Cloneable {
String name;
Address address;
public Object clone() throws CloneNotSupportedException {
return super.clone();
}
}上面这个类实现了 Cloneable 接口,并重写了 clone() 方法,这就是一个典型的浅拷贝实现。当你调用 person.clone() 创建副本后,address 字段还是指向原来的 Address 对象。

常见问题:
address 属性,另一个也会变要实现深拷贝,关键在于递归复制对象内部所有的引用对象。常见的做法有以下几种:
手动逐层克隆: 在每个引用类型的类中都实现 clone() 方法,并在主对象中调用它们。
public Object clone() throws CloneNotSupportedException {
Person p = (Person) super.clone();
p.address = (Address) this.address.clone(); // 深拷贝引用对象
return p;
}序列化方式: 如果对象及所有引用对象都实现了 Serializable 接口,可以通过对象序列化和反序列化来实现深拷贝。
ByteArrayOutputStream bos = new ByteArrayOutputStream(); ObjectOutputStream oos = new ObjectOutputStream(bos); oos.writeObject(original); ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray()); ObjectInputStream ois = new ObjectInputStream(bis); return (T) ois.readObject();
使用第三方库: 比如 Apache Commons Lang 的 SerializationUtils.clone() 或者使用 Gson、Jackson 等 JSON 序列化工具间接实现。
选择建议:
| 特性 | 浅拷贝 | 深拷贝 |
|---|---|---|
| 引用类型字段 | 复制引用地址 | 创建新的引用对象 |
| 修改影响 | 原对象与副本相互影响 | 完全独立 |
| 实现难度 | 简单,直接调用 clone() | 需要递归处理或借助其他机制 |
| 性能 | 快 | 相对较慢,尤其用序列化方式 |
举个例子,假设你有一个用户对象包含地址信息:
基本上就这些。理解清楚浅拷贝和深拷贝的原理和应用场景,才能避免在开发中踩坑。尤其是涉及多线程或者需要保持数据隔离的场景,深拷贝往往更有必要。
以上就是Java中深拷贝与浅拷贝的实现方法与区别详解的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号