在Java中如何将集合深度复制_Java集合深拷贝常见方法

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

Java中集合的深拷贝不能靠简单的赋值或new ArrayList(oldList)实现,因为这些操作只复制了外层容器,内部元素仍指向原对象。要真正实现深拷贝,必须确保集合中的每个元素也被独立复制。

手动遍历 + 元素克隆

适用于元素类型实现了Cloneable接口并重写了clone()方法的情况。注意:clone()默认是浅拷贝,需在元素类中递归处理其成员变量。

  • 确保目标元素类正确实现深克隆逻辑(如对内部数组、引用字段逐个复制)
  • 遍历原集合,对每个元素调用clone(),添加到新集合中
  • 示例:List newList = oldList.stream().map(Person::clone).collect(Collectors.toList());

使用序列化与反序列化

一种通用但稍重

的方案,要求集合内所有元素及其嵌套对象都实现Serializable接口。

  • 将原集合对象写入ByteArrayOutputStream,再通过ObjectInputStream读出
  • 能自动处理多层嵌套和循环引用(前提是可序列化)
  • 性能开销较大,不建议高频调用;注意transient字段不会被复制

借助第三方库(如Apache Commons Lang)

简化代码,避免手写重复逻辑。例如SerializationUtils.clone()底层就是序列化方案,已做封装和异常处理。

  • Maven引入:org.apache.commons:commons-lang3
  • 一行搞定:List newList = SerializationUtils.clone(oldList);
  • 若需更高性能或更灵活控制,可考虑MapStruct(面向DTO转换)或Dozer(已停更,慎用)

不可变集合 + 构造时复制

如果业务允许,用不可变集合(如ImmutableList from Guava)配合构造器传入副本,从源头规避修改风险。

  • ImmutableList.copyOf(oldList)仍是浅拷贝,但返回不可变视图,无法修改
  • 若需真正隔离,应先深拷贝元素,再构建不可变集合:ImmutableList.of(p1Copy, p2Copy, ...)
  • 适合配置类、参数传递等读多写少场景

基本上就这些。选择哪种方式取决于元素是否可序列化、性能要求、项目依赖限制以及是否需要支持复杂嵌套结构。没有银弹,关键是理解“深”在哪里——不是集合本身,而是它装的东西。