当前位置: 首页 > ds >正文

Java中深拷贝与浅拷贝的深入探讨

在Java编程中,对象的拷贝是一个常见且重要的操作,而深拷贝(Deep Copy)浅拷贝(Shallow Copy)是两种主要的拷贝方式。它们在实现原理、应用场景以及性能表现上都有显著的区别。本文将深入探讨这两种拷贝方式,通过代码示例和详细的解释,帮助读者更好地理解和应用它们。

目录

一、浅拷贝与深拷贝的基本概念

1.1 浅拷贝(Shallow Copy)

1.2 深拷贝(Deep Copy)

二、Java中浅拷贝和深拷贝的实现方式

2.1 浅拷贝的实现

2.2 深拷贝的实现

2.3 其他深拷贝实现方式

三、浅拷贝与深拷贝的区别

四、选择拷贝方式的建议

4.1 使用浅拷贝的场景

4.2 使用深拷贝的场景

五、总结


一、浅拷贝与深拷贝的基本概念

1.1 浅拷贝(Shallow Copy)

        浅拷贝是指创建一个新对象,然后将当前对象的非静态字段复制到新对象中。如果字段是值类型的,那么将复制字段的值;如果字段是引用类型的,则复制引用但不复制引用的对象。因此,原始对象和副本对象将引用同一个对象。

特点

  • 基本数据类型的字段会被复制其值。

  • 引用类型字段的引用地址会被复制,但不会复制引用的对象本身,因此原始对象和副本对象共享同一个引用对象。

1.2 深拷贝(Deep Copy)

深拷贝不仅复制对象本身,还会递归地复制对象所引用的所有对象。这意味着深拷贝会为所有引用的对象创建新的副本,从而确保原始对象和副本对象是完全独立的。

特点

  • 所有字段,无论是基本数据类型还是引用类型,都会被独立复制。

  • 修改副本对象中的任何字段都不会影响原始对象,反之亦然。

二、Java中浅拷贝和深拷贝的实现方式

2.1 浅拷贝的实现

在Java中,浅拷贝可以通过Object类的clone()方法实现。默认情况下,clone()方法会进行浅拷贝。

代码示例

class Person implements Cloneable {String name;int age;Address address;  // 引用类型public Person(String name, int age, Address address) {this.name = name;this.age = age;this.address = address;}@Overrideprotected Object clone() throws CloneNotSupportedException {return super.clone();  // 浅拷贝}
}class Address {String city;public Address(String city) {this.city = city;}
}public class Main {public static void main(String[] args) throws CloneNotSupportedException {Address address = new Address("New York");Person p1 = new Person("Alice", 30, address);Person p2 = (Person) p1.clone();  // 浅拷贝// 修改p2的地址p2.address.city = "Los Angeles";// p1的地址也会改变System.out.println(p1.address.city);  // 输出 "Los Angeles"}
}

2.2 深拷贝的实现

深拷贝的实现相对复杂,需要确保递归地复制所有引用类型的字段。

代码示例

class Person implements Cloneable {String name;int age;Address address;  // 引用类型public Person(String name, int age, Address address) {this.name = name;this.age = age;this.address = address;}@Overrideprotected Object clone() throws CloneNotSupportedException {// 首先浅拷贝基本字段Person clonedPerson = (Person) super.clone();// 递归拷贝引用类型的字段clonedPerson.address = (Address) this.address.clone();return clonedPerson;  // 返回完全独立的副本}
}class Address implements Cloneable {String city;public Address(String city) {this.city = city;}@Overrideprotected Object clone() throws CloneNotSupportedException {return super.clone();  // Address的浅拷贝}
}public class Main {public static void main(String[] args) throws CloneNotSupportedException {Address address = new Address("New York");Person p1 = new Person("Alice", 30, address);Person p2 = (Person) p1.clone();  // 深拷贝// 修改p2的地址p2.address.city = "Los Angeles";// p1的地址不会改变System.out.println(p1.address.city);  // 输出 "New York"}
}

2.3 其他深拷贝实现方式

除了使用clone()方法,还可以通过序列化与反序列化实现深拷贝。

代码示例

import java.io.*;class Person implements Serializable {String name;int age;Address address;public Person(String name, int age, Address address) {this.name = name;this.age = age;this.address = address;}/*** 通过序列化与反序列化实现深拷贝* @return 深拷贝后的对象* @throws IOException 如果序列化过程中发生错误* @throws ClassNotFoundException 如果反序列化过程中找不到类*/public Person deepCopy() throws IOException, ClassNotFoundException {// 创建一个字节输出流,用于存储序列化后的对象数据ByteArrayOutputStream baos = new ByteArrayOutputStream();// 创建一个对象输出流,用于将对象写入字节输出流ObjectOutputStream oos = new ObjectOutputStream(baos);// 将当前对象序列化并写入字节输出流oos.writeObject(this);// 将字节输出流转换为字节输入流,用于反序列化ByteArrayInputStream bais = new ByteArrayInputStream(baos.toByteArray());// 创建一个对象输入流,用于从字节输入流中读取对象ObjectInputStream ois = new ObjectInputStream(bais);// 从对象输入流中读取对象,完成反序列化,得到深拷贝后的对象return (Person) ois.readObject();}
}class Address implements Serializable {String city;public Address(String city) {this.city = city;}
}public class Main {public static void main(String[] args) throws IOException, ClassNotFoundException {Address address = new Address("New York");Person p1 = new Person("Alice", 30, address);// 调用深拷贝方法Person p2 = p1.deepCopy();// 修改p2的地址p2.address.city = "Los Angeles";// p1的地址不会改变System.out.println(p1.address.city);  // 输出 "New York"}
}

三、浅拷贝与深拷贝的区别

对比点浅拷贝深拷贝
复制对象层次只复制第一层对象,引用类型字段共享同一对象递归复制所有对象及其引用类型字段
引用共享内部引用类型共享同一内存地址内部对象完全独立,不共享任何引用
适用场景适用于简单对象,没有深层次引用适用于复杂对象结构,深层对象独立

四、选择拷贝方式的建议

4.1 使用浅拷贝的场景

  1. 对象结构简单,没有嵌套引用类型。

  2. 性能要求高。

  3. 明确知道只需要第一层拷贝。

4.2 使用深拷贝的场景

  1. 对象结构复杂,有多层嵌套。

  2. 需要完全独立的副本。

  3. 不关心性能开销。

五、总结

        在Java中,浅拷贝和深拷贝是两种重要的对象拷贝方式。浅拷贝通过clone()方法实现,简单且性能高,但只复制对象的第一层,引用类型字段共享同一对象。深拷贝则通过递归复制所有引用类型的字段,确保新对象与原对象完全独立,但实现复杂且性能较低。在实际开发中,应根据对象的结构和需求选择合适的拷贝方式。


希望本文能帮助你更好地理解Java中的深拷贝与浅拷贝。

http://www.xdnf.cn/news/3803.html

相关文章:

  • C++类_虚基类
  • IDEA快速上手Maven项目:模板选择 + 多模块拆分
  • Spring Boot 微服务打包为 Docker 镜像并部署到镜像仓库实战案例
  • 合成复用原则(CRP)
  • IDEA回滚代码操作
  • Windows下调试WebRTC源码
  • BOSS的收入 - 华为OD机试(A卷,C++题解)
  • 昇腾的昇思MindSpore是什么?跟TensorFlow/PyTorch 等第三方框架有什么区别和联系?【浅谈版】
  • c++ 二级指针 vs 指针引用
  • 小土堆pytorch数据加载概念以及实战
  • 【Tauri2】37——后端处理invoke
  • SVM实战:从理论到鸢尾花数据集的分类可视化
  • skynet中的client.socket库函数详解
  • WebRTC 服务器之Janus架构分析
  • 大模型开发的环节
  • AIGC算力消耗白皮书:Stable Diffusion vs Midjourney的架构成本差异
  • 头歌数据库课程实验(索引与数据库完整性)
  • 从零认识阿里云OSS:云原生对象存储的核心价值
  • 解析机器人 2.0.2 | 支持超过50种短视频平台的链接解析,无水印提取,多功能下载工具
  • 华为OD机试真题——智能驾驶(2025A卷:200分)Java/python/JavaScript/C/C++/GO最佳实现
  • 什么是函数重载?
  • MySQL 空值处理函数对比:IFNULL、COALESCE 和 NULLIF
  • 《Linux macOS :GCC升级方法》
  • 私人医生通过AI分析基因数据,是否有权提前告知癌症风险?
  • 【AI面试准备】XMind拆解业务场景识别AI赋能点
  • QML图像提供器 (Image Provider)
  • 【Ansible自动化运维实战:从Playbook到负载均衡指南】
  • 【算法基础】插入排序算法 - JAVA
  • 怎样增加AI对话的拟人化和增加同理心
  • WEB前端小练习——记事本