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

直接用 new ArrayList(original) 最安全
绝大多数场景下,只要原始集合是 ArrayList、LinkedList 这类支持随机访问或迭代的实现,用构造函数复制就是最简洁、最不容易出错的方式。它创建的是浅拷贝,元素引用不变,但集合容器本身是新的。
常见错误是误以为 list1 = list2 是复制——其实只是引用赋值,后续对 list2 的 add、remove 会直接影响 list1。
- 适用于所有实现了
Collection接口的集合(HashSet、LinkedHashSet同理) - 不触发泛型擦除问题,编译期类型安全
- 性能好:内部通常调用
Arrays.copyOf或批量add,比循环手动 add 快
Collection.addAll() 适合已有目标集合需追加时
如果目标集合已存在,且你只想把源集合所有元素“加进去”,而不是新建一个副本,addAll() 是正确选择。但它不是拷贝操作,而是合并行为——这点常被混淆。
典型误用:先 new ArrayList(),再 addAll(),不如直接走构造函数一步到位。
立即学习“Java免费学习笔记(深入)”;
- 必须确保目标集合可修改(比如不是
Collections.unmodifiableList()包装过的) - 若目标集合已有元素,结果是叠加而非替换
- 返回
boolean表示是否发生了结构变化,调试时可用来判断源集合是否为空
深拷贝必须自己处理,Java 没有通用方案
Java 集合默认只做浅拷贝:new ArrayList(original) 新建了列表对象,但里面每个元素仍是原对象的引用。如果元素是可变对象(如自定义 User 类),改副本里的 user.setName("A") 会影响原集合中的同一实例。
没有银弹。常见做法:
- 元素类实现
Cloneable并重写clone(),然后遍历调用 —— 但clone()本身有缺陷,不推荐 - 用构造函数传参方式重建(如
new User(user.getName(), user.getAge())),最可控 - 借助 Jackson/Gson 序列化反序列化(适合 POJO 且无循环引用),但有性能和兼容性成本
ListdeepCopy = original.stream() .map(u -> new User(u.getName(), u.getAge())) .collect(Collectors.toList());
Arrays.asList() 返回的集合不能直接复制
这个方法返回的是 Arrays 内部的静态嵌套类,它不支持 add、remove、clear 等结构性修改操作,调用会抛 UnsupportedOperationException。很多人试图用它做“快捷初始化”,然后想复制它,却卡在第一步。
正确做法:把它当作只读输入,立刻包进真正的集合里再操作。
- 错:
List→ 运行时报错list = Arrays.asList("a", "b"); list.add("c"); - 对:
Listlist = new ArrayList(Arrays.asList("a", "b")); - 注意:
Arrays.asList()对基本类型数组(如int[])无效,它会把整个数组当做一个元素










