Java中的深拷贝与浅拷贝

ops/2025/3/16 19:57:22/

在Java中,深拷贝和浅拷贝是两种不同的对象复制方式,主要区别在于它们如何处理对象内部的引用类型字段。

浅拷贝(Shallow Copy)

浅拷贝创建一个新对象,并将原对象的字段值复制到新对象中。如果字段是基本类型,则直接复制值;如果字段是引用类型,则复制引用(即内存地址),因此新对象和原对象共享相同的引用类型字段。

实现方式:

  • 实现 Cloneable 接口并重写 clone() 方法。
  • 使用工具类(如 BeanUtils.copyProperties)。
class Person implements Cloneable {String name;Address address;public Person(String name, Address address) {this.name = name;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("Beijing");Person person1 = new Person("Alice", address);Person person2 = (Person) person1.clone();System.out.println(person1.address == person2.address); // true,共享同一个Address对象}
}

深拷贝(Deep Copy)

深拷贝创建一个新对象,并递归地复制原对象的所有字段。如果字段是引用类型,则创建该字段的新副本,而不是共享引用。因此,新对象和原对象完全独立。
实现方式:

  • 手动实现递归复制。
  • 使用序列化(如 Serializable 接口)。
  • 使用第三方库(如 Apache Commons Lang 的 SerializationUtils)。
  • 如果对象包含循环引用,深拷贝需要特别处理,否则可能导致栈溢出。
  • 序列化方式实现深拷贝要求所有相关类都实现 Serializable 接口。
import java.io.*;class Person implements Serializable {String name;Address address;public Person(String name, Address address) {this.name = name;this.address = address;}public Person deepCopy() throws IOException, ClassNotFoundException {// 使用序列化实现深拷贝ByteArrayOutputStream bos = new ByteArrayOutputStream();ObjectOutputStream oos = new ObjectOutputStream(bos);oos.writeObject(this);ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());ObjectInputStream ois = new ObjectInputStream(bis);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("Beijing");Person person1 = new Person("Alice", address);Person person2 = person1.deepCopy();System.out.println(person1.address == person2.address); // false,Address对象是独立的}
}

对比

特性浅拷贝深拷贝
基本类型字段复制值复制值
引用类型字段复制引用(共享同一对象)递归复制(创建新对象)
实现复杂度简单复杂
性能较高较低
适用场景对象内部没有引用类型字段或共享引用无影响需要完全独立的对象副本

拓展知识

Java 中的 clone() 方法

  • clone() 是 Object 类的一个方法,用于创建对象的副本。
  • 默认的 clone() 方法是浅拷贝。
  • 使用 clone() 需要满足以下条件:
    • 类必须实现 Cloneable 接口(标记接口,没有方法)。
    • 重写 clone() 方法,并将其访问修饰符改为 public。
  • 注意:clone() 方法不会调用构造函数。
class Person implements Cloneable {String name;int age;public Person(String name, int age) {this.name = name;this.age = age;}@Overridepublic Person clone() {try {return (Person) super.clone();} catch (CloneNotSupportedException e) {throw new RuntimeException(e);}}
}

序列化实现深拷贝

通过序列化和反序列化可以实现深拷贝。但要求所有相关类都实现 Serializable 接口。

  • 优点:简单易用,适合复杂对象图的深拷贝。
  • 缺点:性能较低,且要求所有字段都可序列化。
import java.io.*;class DeepCopyUtil {public static <T extends Serializable> T deepCopy(T object) {try (ByteArrayOutputStream bos = new ByteArrayOutputStream();ObjectOutputStream oos = new ObjectOutputStream(bos)) {oos.writeObject(object);try (ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());ObjectInputStream ois = new ObjectInputStream(bis)) {return (T) ois.readObject();}} catch (IOException | ClassNotFoundException e) {throw new RuntimeException("Deep copy failed", e);}}
}

深拷贝与不可变对象
如果对象是不可变的(如 String、Integer 等),则浅拷贝和深拷贝的效果相同。不可变对象的值无法被修改,因此共享引用是安全的。

String s1 = "Hello";
String s2 = s1; // 浅拷贝,但因为是不可变对象,所以安全

深拷贝的性能问题
深拷贝需要递归复制整个对象图,可能会消耗较多内存和 CPU 资源。如果对象图非常大或嵌套层级很深,深拷贝可能会导致性能问题。
优化方法:

  • 使用懒加载(Lazy Copy):只有在修改时才复制对象。
  • 使用对象池或缓存机制。

深拷贝与线程安全

如果多个线程共享同一个对象,浅拷贝可能导致线程安全问题。深拷贝可以避免线程安全问题,因为每个线程操作的是独立的对象副本。


http://www.ppmy.cn/ops/166300.html

相关文章

本地知识库RAG总结

目录 RAG流程: 知识库的要求&#xff1a; 知识抽取&#xff1a; 知识存储: 向量化: 知识检索: 应用客户端: RAG智能问答应用几个痛点&#xff1a; 如何提升召回率改进思路&#xff1a; 如何提升回答专业性&#xff1a; RAG评测&#xff1a; 总结&#xff1a; 参考…

【实战-解决方案】Webpack 打包后很多js方法报错:not defined

问题分析 在不打包的情况下&#xff0c;方法&#xff08;如 checkLoginStatus、filterSites、initProgressBar 等&#xff09;可以正常运行&#xff0c;而经过 Webpack 打包后报 is not defined 错误&#xff0c;通常有以下几个可能的原因&#xff1a; 全局变量丢失 在 Webpac…

2024年第十五届蓝桥杯软件C/C++大学A组——五子棋对弈

蓝桥杯原题&#xff1a; 题目描述&#xff1a; “在五子棋的对弈中&#xff0c;友谊的小船说翻就翻&#xff1f; ” 不&#xff01;对小蓝和小桥来说&#xff0c;五子棋不仅是棋盘上的较量&#xff0c;更是心与心之间的沟通。这两位挚友秉承着 “ 友谊第一&#xff0c;比赛第二…

Ollama+OpenManus详细部署实战

一,OpenManus介绍 一个专注于基于强化学习(RL,例如 GRPO)的方法来优化大语言模型(LLM)智能体的开源项目,由来自UIUC 和 OpenManus 的研究人员合作开发。 二,安装 创建新的 conda 环境:conda create -n OpenManus python=3.12 conda activate OpenManus克隆仓库:git c…

面试系列|蚂蚁金服技术面【2】

今天继续分享一下蚂蚁金服的 Java 后端开发岗位真实社招面经&#xff0c;复盘面试过程中踩过的坑&#xff0c;整理面试过程中提到的知识点&#xff0c;希望能给正在准备面试的你一些参考和启发&#xff0c;希望对你有帮助&#xff0c;愿你能够获得心仪的 offer ! 第一轮面试完…

Liunx启动kafka并解决kafka时不时挂掉的问题

kafka启动步骤 先启动zookeeper&#xff0c;启动命令如下 nohup ./zookeeper-server-start.sh /home/kafka/kafka/config/zookeeper.properties > /home/kafka/kafka/zookeeper.log 2>&1 &再启动kafka&#xff0c;启动命令如下 nohup ./kafka-server-start.sh…

redis部署架构

一、redis多实例部署 实例1 安装目录&#xff1a;/app/6380 数据目录&#xff1a;/app/6380/data 实例2 安装目录&#xff1a;/app/6381 数据目录&#xff1a;/app/6381/data 1、创建实例安装目录 2、拷贝实例的配置文件 3、编辑实例的配置文件 第…

自定义tiptap插件

本文为开发开源项目的真实开发经历&#xff0c;感兴趣的可以来给我的项目点个star&#xff0c;谢谢啦~ 具体博文介绍&#xff1a; 开源&#xff5c;Documind协同文档&#xff08;接入deepseek-r1、支持实时聊天&#xff09;Documind &#x1f680; 一个支持实时聊天和接入 - 掘…