在Java中如何复制集合_Java集合拷贝方法解析

最安全的集合复制方式是直接使用 new ArrayList(original),它创建浅拷贝、类型安全、性能好,适用于所有 Collection 实现;需注意 Arrays.asList() 返回不可变列表,深拷贝须业务自行实现。

直接用 new ArrayList(original) 最安全

绝大多数场景下,只要原始集合是 ArrayListLinkedList 这类支持随机访问或迭代的实现,用构造函数复制就是最简洁、最不容易出错的方式。它创建的是浅拷贝,元素引用不变,但集合容

器本身是新的。

常见错误是误以为 list1 = list2 是复制——其实只是引用赋值,后续对 list2addremove 会直接影响 list1

  • 适用于所有实现了 Collection 接口的集合(HashSetLinkedHashSet 同理)
  • 不触发泛型擦除问题,编译期类型安全
  • 性能好:内部通常调用 Arrays.copyOf 或批量 add,比循环手动 add 快

Collection.addAll() 适合已有目标集合需追加时

如果目标集合已存在,且你只想把源集合所有元素“加进去”,而不是新建一个副本,addAll() 是正确选择。但它不是拷贝操作,而是合并行为——这点常被混淆。

典型误用:先 new ArrayList(),再 addAll(),不如直接走构造函数一步到位。

  • 必须确保目标集合可修改(比如不是 Collections.unmodifiableList() 包装过的)
  • 若目标集合已有元素,结果是叠加而非替换
  • 返回 boolean 表示是否发生了结构变化,调试时可用来判断源集合是否为空

深拷贝必须自己处理,Java 没有通用方案

Java 集合默认只做浅拷贝:new ArrayList(original) 新建了列表对象,但里面每个元素仍是原对象的引用。如果元素是可变对象(如自定义 User 类),改副本里的 user.setName("A") 会影响原集合中的同一实例。

没有银弹。常见做法:

  • 元素类实现 Cloneable 并重写 clone(),然后遍历调用 —— 但 clone() 本身有缺陷,不推荐
  • 用构造函数传参方式重建(如 new User(user.getName(), user.getAge())),最可控
  • 借助 Jackson/Gson 序列化反序列化(适合 POJO 且无循环引用),但有性能和兼容性成本
List deepCopy = original.stream()
    .map(u -> new User(u.getName(), u.getAge()))
    .collect(Collectors.toList());

Arrays.asList() 返回的集合不能直接复制

这个方法返回的是 Arrays 内部的静态嵌套类,它不支持 addremoveclear 等结构性修改操作,调用会抛 UnsupportedOperationException。很多人试图用它做“快捷初始化”,然后想复制它,却卡在第一步。

正确做法:把它当作只读输入,立刻包进真正的集合里再操作。

  • 错: List list = Arrays.asList("a", "b"); list.add("c"); → 运行时报错
  • 对: List list = new ArrayList(Arrays.asList("a", "b"));
  • 注意:Arrays.asList() 对基本类型数组(如 int[])无效,它会把整个数组当做一个元素
浅拷贝够用是常态,但一旦元素含可变状态,深拷贝逻辑就得落在业务代码里——没人能替你决定哪些字段要复制、哪些可以共享。