合并List首选addAll(),需目标集合可变;Set合并推荐构造器+addAll();Stream.concat()仅支持两流且不处理null;Guava Iterables.concat()为懒视图,需转实体集合。

用 addAll() 合并 List 是最常用也最直接的方式
多数场景下,你只是想把几个 List 的元素拼成一个新列表,addAll() 就够用了。它不改变原集合,但要求目标集合是可变的(比如 new ArrayList<>()),不能对 Arrays.asList() 或 Collections.unmodifiableList() 调用。
常见错误现象:UnsupportedOperationException —— 本质是底层用了不可变集合或固定大小数组。
- 推荐写法:
List<String> result = new ArrayList<>(); list1.addAll(list2); list1.addAll(list3);
- 如果要保留原集合不变,先初始化空
ArrayList再逐个addAll() - 注意:
addAll()是顺序追加,重复元素不会去重
合并 Set 时优先用构造器,避免手动 add
合并多个 Set,最简洁安全的方式是用构造器:new HashSet<>(set1) 然后 addAll(set2),而不是循环 add()。前者利用了哈希表批量插入的内部优化,后者在大集合时性能明显下降。
使用场景:去重合并、权限集合合并、标签聚合等。
- 别写:
Set<String> result = new HashSet<>(); for (String s : set1) result.add(s); for (String s : set2) result.add(s);
- 推荐写:
Set<String> result = new HashSet<>(set1); result.addAll(set2);
- 如果需要保持插入顺序,用
LinkedHashSet替代HashSet
Stream.concat() 适合函数式风格,但要注意 null 和类型一致
Stream.concat() 可以链式合并两个流,再用 collect() 转回集合。但它只接受两个参数,合并三个及以上得嵌套调用,可读性会下降;而且任意一个输入为 null 就抛 NullPointerException。
参数差异:concat() 要求两个流元素类型完全一致(泛型擦除后也不能混 String 和 Integer)。
- 正确示例:
List<String> result = Stream.concat(list1.stream(), list2.stream()) .collect(Collectors.toList()); - 错误风险点:如果
list1或list2是null,必须提前判空 - 合并三个以上建议改用
Stream.of(list1, list2, list3).flatMap(List::stream)
Guava 的 Iterables.concat() 更轻量,但需引入依赖
如果你已经在用 Guava,Iterables.concat() 不立即执行合并,返回的是一个懒视图(lazy view),只有遍历时才拉取数据。内存友好,适合超大集合或 IO 流式场景。
容易踩的坑:Iterables.concat() 返回的不是 List 或 Set,不能直接当集合用;若需实体集合,仍得调用 Lists.newArrayList() 或 Sets.newHashSet() 包一层。
- 示例:
Iterable<String> merged = Iterables.concat(list1, list2, list3); List<String> realList = Lists.newArrayList(merged);
- 注意:它不自动去重,也不保证顺序(取决于输入迭代器行为)
- 没有 Guava 的项目别为了这个功能专门引入,得不偿失