最常用且安全的浅拷贝方式是用构造方法:new ArrayList(originalList) 或 new LinkedList(originalList),但需注意判空、类型适配及不可变集合的“假拷贝”风险。

直接用构造方法复制 ArrayList 或 LinkedList
大多数时候,你只需要一个浅拷贝——新集合和原集合元素引用相同,但集合容器本身独立。这时候最简单的方式就是用目标集合的构造方法传入原集合:
-
new ArrayList(originalList)是最常用、最安全的写法,适用于任何Collection子类 -
new LinkedList(originalList)同理,但注意它内部是链表结构,随机访问性能差,别只因“想复制”就盲目换类型 - 如果
originalList是null,会抛NullPointerException,务必提前判空或用Objects.requireNonNull
用 Collections.copy() 时必须注意容量匹配
Collections.copy() 不创建新集合,而是把源集合元素逐个赋值到目标集合已有位置。它要求目标集合长度 ≥ 源集合长度,否则抛 IndexOutOfBoundsException:
- 目标集合必须已初始化且有足够容量,例如:
Listdest = new ArrayList(Collections.nCopies(src.size(), null)) - 它不会扩容,也不会清空原内容——如果
dest原来有 10 个元素,只复制了前 5 个,后 5 个保留原值 - 实际项目中极少用它,除非你明确要复用某段已分配内存的列表(比如池化场景),否则容易出错
深拷贝不能靠集合构造方法解决
构造方法只复制引用,若集合里存的是可变对象(如 HashMap、自定义 POJO),修改副本里的对象仍会影响原集合中的对应对象:
- 真需要深拷贝,得逐个调用元素的克隆逻辑,比如:
list.stream().map(item -> item.clone()).collect(Collectors.toList()),前提是item实现了Cloneable且重写了clone() - 更稳妥的做法是用序列化(如
ObjectOutputStream+ByteArrayInputStream),但要求所有元素都实现Serializable,且注意 transient 字段不会被复制 - JSON 序列化(如 Jackson)也是一种选择,但会引入额外依赖、性能开销大,且对循环引用、特殊类型(如
Date)需额外配置
不可变集合复制要小心“假拷贝”
如果你用的是 ImmutableList.copyOf()(Guava)或 List.of()(Java 9+),它们返回的是不可变视图,底层可能共享引用:
立即学习“Java免费学习笔记(深入)”;
-
ImmutableList.copyOf(original)如果original本身已是不可变实现,可能直接返回它,而非新建对象 -
List.copyOf()(Java 10+)也是类似逻辑:若参数已是不可变List,则不复制;否则才新建 - 这意味着你以为“复制了”,其实两个变量指向同一实例,
==可能为true——调试时容易误判
真正要复制集合,先想清楚:要的是容器独立?还是元素也隔离?是否允许 null?有没有并发修改风险?这些细节比“怎么写一行代码”更重要。









