浅拷贝复制对象及其基本类型字段值,引用类型仅复制地址;深拷贝递归复制所有引用对象,形成独立副本。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中深拷贝与浅拷贝的实现方法与区别详解的详细内容,更多请关注创想鸟其它相关文章!
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/141457.html
微信扫一扫
支付宝扫一扫