0

0

详解Java中的深拷贝与浅拷贝实现_Cloneable接口与序列化拷贝对比

P粉602998670

P粉602998670

发布时间:2026-02-27 06:43:15

|

364人浏览过

|

来源于php中文网

原创

浅拷贝只复制引用是因为object.clone()默认逐字段复制,基本类型值照搬,引用类型仅复制地址;深拷贝用序列化虽可避免引用共享,但要求所有字段可序列化且无运行时依赖,性能较差。

详解java中的深拷贝与浅拷贝实现_cloneable接口与序列化拷贝对比

浅拷贝为什么只复制了引用?

因为 Object.clone() 默认行为就是逐字段复制:基本类型值照搬,引用类型只复制地址。对象本身没被重建,所以修改副本里的数组或集合,原对象跟着变。

常见错误现象:clone() 后修改 list.add(),原对象的 list 也多了一项;person.getAddress().setCity("Beijing") 同时改了原 person 和克隆体。

  • 必须手动重写 clone(),并为每个可变引用字段调用其自身的 clone() 或新建对象
  • 类要实现 Cloneable 接口,否则抛 CloneNotSupportedException
  • 注意子类继承父类 clone() 时,父类字段是否已正确处理——容易漏掉父类里新增的引用字段

深拷贝用序列化真的安全吗?

序列化拷贝(通过 ObjectOutputStream / ObjectInputStream)能绕过引用共享问题,但前提是所有字段所属类型都可序列化,且不依赖 JVM 运行时状态(比如线程、Socket、数据库连接等)。

使用场景:配置对象、DTO、纯数据载体类;不适合含回调、监听器、Spring 代理对象或 ThreadLocal 字段的类。

立即学习Java免费学习笔记(深入)”;

Genspark
Genspark

Genspark 是一款创新的 AI 搜索引擎,致力于提供比传统搜索引擎更高效、准确和无偏见的信息获取方式。

下载
  • 类必须实现 Serializable,且所有非 transient 引用字段也要可序列化
  • statictransient 字段不会被序列化,拷贝后为默认值(null / 0 / false
  • 性能比 clone() 差不少,尤其对象深、字段多时,序列化/反序列化有明显开销
  • 某些 JDK 版本对匿名内部类序列化支持不稳定,可能抛 NotSerializableException

第三方库(如 Apache Commons Lang)怎么选?

SerializationUtils.clone() 底层还是走序列化,只是封装了异常和流操作;而 BeanUtils.cloneBean() 是基于反射的浅拷贝,不解决嵌套引用问题——名字带 “clone” 不代表是深拷贝。

容易踩的坑:SerializationUtils.clone() 对不可序列化字段静默失败(返回 null),不抛异常;BeanUtils 遇到同名不同类型的属性直接跳过,也不报错。

  • 确认你用的是 org.apache.commons.lang3.SerializationUtils.clone(),不是老版 commons-lang 的同名方法
  • 若字段含 java.time.LocalDateTime 等新时间类,需确保 JDK ≥ 8 且序列化机制兼容(它们实现了 Serializable
  • 不要在循环里频繁调用,避免反复创建 ByteArrayOutputStreamObjectInputStream

自定义深拷贝要不要手写递归?

手写递归拷贝(比如遍历字段 + 判断类型 + 分支处理)理论上最可控,但实际项目中几乎没人这么做——太容易漏字段、错判类型、陷入循环引用(A→B→A)导致栈溢出。

真正需要定制逻辑的场景极少,比如字段含加密上下文、缓存句柄、或需按业务规则跳过某些字段。这时候更稳妥的做法是:只对关键字段重写 clone(),其余走默认序列化,再组合。

  • 如果真要递归,务必加访问标记(如 Set<object></object> 记录已拷贝对象)防循环引用
  • 优先考虑 copy constructor(构造函数传入原对象)或 builder 模式,语义清晰、IDE 友好、调试方便
  • 注意泛型擦除问题:运行时无法知道 List<string></string>List<integer></integer> 的区别,反射取值易出 ClassCastException
深拷贝没有银弹,关键看字段构成和运行环境。序列化看着省事,但一碰到不可序列化字段或性能敏感路径,就得退回去补 clone();而手写递归看似彻底,维护成本和出错概率反而更高。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
spring框架介绍
spring框架介绍

本专题整合了spring框架相关内容,想了解更多详细内容,请阅读专题下面的文章。

146

2025.08.06

Java Spring Security 与认证授权
Java Spring Security 与认证授权

本专题系统讲解 Java Spring Security 框架在认证与授权中的应用,涵盖用户身份验证、权限控制、JWT与OAuth2实现、跨站请求伪造(CSRF)防护、会话管理与安全漏洞防范。通过实际项目案例,帮助学习者掌握如何 使用 Spring Security 实现高安全性认证与授权机制,提升 Web 应用的安全性与用户数据保护。

83

2026.01.26

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

248

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

906

2024.03.01

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1679

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

506

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

2318

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

41

2026.01.19

html5播放器怎么用
html5播放器怎么用

本合集全面介绍HTML5播放器的使用方法,涵盖基础语法、自定义控制、兼容性处理及实战示例。阅读专题下面的文章了解更多详细内容。

0

2026.02.27

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 3.9万人学习

C# 教程
C# 教程

共94课时 | 10.2万人学习

Java 教程
Java 教程

共578课时 | 72.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号