Java集合克隆:深拷贝与浅拷贝区别详解
时间:2026-04-04 08:14:12 101浏览 收藏
Java集合克隆中,浅拷贝(如ArrayList.clone()、构造函数或Collections.copy())仅复制容器引用,元素对象仍共享,修改可变元素会波及原集合;而深拷贝需手动逐元素clone、确保类正确实现Cloneable并递归处理嵌套对象,或借助序列化(要求全链路Serializable但性能差、限制多),本质上深拷贝不是技术调用问题,而是开发者对状态隔离边界的主动设计与权衡——理解这一点,才能避开“new了就安全”的认知陷阱,在真实业务中做出可靠选择。

ArrayList.clone() 返回的是浅拷贝,不是深拷贝
ArrayList.clone() 只复制了集合本身(即内部的 Object[] 数组引用),但数组里的每个元素对象仍指向原对象。如果元素是可变对象(比如 HashMap、自定义类实例),修改副本里的元素会同步影响原集合。
- 常见错误现象:
list2 = (ArrayList) list1.clone(); list2.get(0).put("k", "v");结果list1.get(0)也变了 - 适用场景:元素全是不可变类型(
String、Integer)时,浅拷贝够用 - 注意:
clone()返回Object,必须强制转型,且要求元素类实现Cloneable才能安全调用其clone()
手动实现深拷贝:逐个 clone 元素再 new ArrayList
没有银弹,得自己控制“深”的边界——你要 clone 到哪一层?通常指复制元素本身,而非只复制引用。
- 使用场景:元素是自定义类,且该类已正确重写
clone()方法并声明implements Cloneable - 实操建议:
ArrayList<MyObj> deepCopy = new ArrayList<>(); for (MyObj obj : original) { deepCopy.add(obj.clone()); } - 坑点:如果
MyObj.clone()内部没递归 clone 字段(比如它有个Map字段),那还是浅的——深不深,取决于你写的> clone()逻辑
用构造函数或 Collections.copy() 都只是浅拷贝
new ArrayList(otherList) 和 Collections.copy(dest, src) 都只做引用复制,和 clone() 行为一致,别被“new”字眼误导。
- 错误认知:“new 一个新 ArrayList 就等于深拷贝”——错,它只新建了容器,没碰里面的东西
Collections.copy()要求目标集合已有足够容量,否则抛IndexOutOfBoundsException,且仍是逐引用赋值- 性能影响:这些方式都 O(n) 时间,但内存上不额外深拷贝对象,所以比真深拷贝省空间
序列化方式能绕过 clone 约束,但代价高且有局限
把对象写成字节流再读回来,天然达成深拷贝效果,适合不想改源码又无法控制元素类是否实现 Cloneable 的情况。
- 前提:所有元素类及其字段都必须实现
Serializable,且不含transient关键数据(否则丢失) - 示例代码片段:
ByteArrayOutputStream bos = new ByteArrayOutputStream(); ObjectOutputStream oos = new ObjectOutputStream(bos); oos.writeObject(original); oos.close(); ObjectInputStream ois = new ObjectInputStream(new ByteArrayInputStream(bos.toByteArray())); ArrayList<?> deepCopy = (ArrayList<?>) ois.readObject();
- 坑点:静态字段不参与序列化;Lambda、内部类可能因隐式持外层引用导致
NotSerializableException;性能差,GC 压力大,纯当保底方案
ThreadLocal、Socket 或带 native 句柄的对象,照样崩。今天带大家了解了的相关知识,希望对你有所帮助;关于文章的技术知识我们会一点点深入介绍,欢迎大家关注golang学习网公众号,一起学习编程~
相关阅读
更多>
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
最新阅读
更多>
-
224 收藏
-
330 收藏
-
436 收藏
-
374 收藏
-
153 收藏
-
201 收藏
-
229 收藏
-
110 收藏
-
392 收藏
-
356 收藏
-
384 收藏
-
160 收藏
课程推荐
更多>
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习