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

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中的深拷贝与浅拷贝。

相关文章:

  • C++抽象基类三重防线:纯虚函数与保护构造的深度实践
  • springAop代理责任链模式源码解析
  • 《解锁GCC版本升级:开启编程新世界大门》
  • Python蓝桥杯真题代码
  • 工作记录 2015-06-01
  • 数据库介绍以及windows下mysql安装
  • vector和string的迭代器
  • BG开发者日志505:项目总体情况
  • PowerPC架构详解:定义、应用及特点
  • 软件管理(安装方式)
  • MCP 探索:MCP 集成的相关网站 Smithery、PulseMCP 等
  • MySQL安装完全指南:从零开始到配置优化(附避坑指南)
  • 【Python生成器与迭代器】核心原理与实战应用
  • 【Python实战】飞机大战
  • 吾爱出品 [Windows] EndNote 21.5.18513 汉化补丁
  • 【进阶】C# 委托(Delegate)知识点总结归纳
  • Rotary Positional Embedding
  • QT6 源(72):阅读与注释单选框这个类型的按钮 QRadioButton,及各种属性验证,
  • 存在重复元素II(简单)
  • Three.js在vue中的使用(二)-加载、控制
  • 张建华评《俄国和法国》|埃莲娜·唐科斯的俄法关系史研究
  • “五一”假期首日跨区域人员流动预计超3.4亿人次
  • 同日哑火丢冠,双骄的下山路,手牵手一起走
  • 美乌矿产协议签署被曝“临门一脚”时生变,美方提附加条件
  • 南京106亿元成交19宗涉宅地块:建邺区地块楼面单价重回4.5万元
  • 美参议院通过新任美国驻华大使任命,外交部回应