Java集合深拷贝需复制内部元素而非仅容器,方法包括:1. 手动遍历+克隆(要求元素实现Cloneable并重写clone);2. 序列化反序列化(要求Serializable,自动处理嵌套但性能低);3. 第三方库如SerializationUtils.clone();4. 不可变集合配合元素深拷贝。

Java中集合的深拷贝不能靠简单的赋值或new ArrayList(oldList)实现,因为这些操作只复制了外层容器,内部元素仍指向原对象。要真正实现深拷贝,必须确保集合中的每个元素也被独立复制。
手动遍历 + 元素克隆
适用于元素类型实现了Cloneable接口并重写了clone()方法的情况。注意:clone()默认是浅拷贝,需在元素类中递归处理其成员变量。
- 确保目标元素类正确实现深克隆逻辑(如对内部数组、引用字段逐个复制)
- 遍历原集合,对每个元素调用
clone(),添加到新集合中 - 示例:
ListnewList = oldList.stream().map(Person::clone).collect(Collectors.toList());
使用序列化与反序列化
一种通用但稍重的方案,要求集合内所有元素及其嵌套对象都实现Serializable接口。
- 将原集合对象写入
ByteArrayOutputStream,再通过ObjectInputStream读出 - 能自动处理多层嵌套和循环引用(前提是可序列化)
- 性能开销较大,不建议高频调用;注意
transient字段不会被复制
借助第三方库(如Apache Commons Lang)
简化代码,避免手写重复逻辑。例如SerializationUtils.clone()底层就是序列化方案,已做封装和异常处理。
立即学习“Java免费学习笔记(深入)”;
- Maven引入:
org.apache.commons:commons-lang3 - 一行搞定:
ListnewList = SerializationUtils.clone(oldList); - 若需更高性能或更灵活控制,可考虑
MapStruct(面向DTO转换)或Dozer(已停更,慎用)
不可变集合 + 构造时复制
如果业务允许,用不可变集合(如ImmutableList from Guava)配合构造器传入副本,从源头规避修改风险。
-
ImmutableList.copyOf(oldList)仍是浅拷贝,但返回不可变视图,无法修改 - 若需真正隔离,应先深拷贝元素,再构建不可变集合:
ImmutableList.of(p1Copy, p2Copy, ...) - 适合配置类、参数传递等读多写少场景
基本上就这些。选择哪种方式取决于元素是否可序列化、性能要求、项目依赖限制以及是否需要支持复杂嵌套结构。没有银弹,关键是理解“深”在哪里——不是集合本身,而是它装的东西。










