
在Java中,不可变集合一旦创建就不能被修改。如果你需要基于现有的不可变集合创建一个新的、包含更多元素的集合,直接添加元素是不允许的。 幸运的是,Java 8引入的Stream API 提供了一种优雅的解决方案。
让我们考虑以下场景:你有一个不可变集合 Set
以下是如何使用Stream API实现此目的:
import java.util.Set;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import java.util.function.Function;
public class ImmutableSetExample {
public static void main(String[] args) {
Set s = Set.of("a", "b", "c");
// 方法一:使用 Set.of() 创建一个包含单个元素的 Set,然后合并
Set t1 = Stream.of(s, Set.of("d"))
.flatMap(Set::stream)
.collect(Collectors.toUnmodifiableSet());
System.out.println("方法一: " + t1); // 输出: [d, c, b, a] (顺序可能不同)
// 方法二:直接将 Stream 合并
Set t2 = Stream.of(s.stream(), Stream.of("d"))
.flatMap(Function.identity())
.collect(Collectors.toUnmodifiableSet());
System.out.println("方法二: " + t2); // 输出: [d, c, b, a] (顺序可能不同)
}
} 代码解释:
-
Stream.of(s, Set.of("d")) (方法一) / Stream.of(s.stream(), Stream.of("d")) (方法二): 这部分代码创建了一个包含两个元素的Stream。 在方法一中,每个元素都是一个Set
,分别是原始集合 s 和包含 "d" 的新集合。在方法二中,每个元素都是一个Stream ,分别是原始集合 s 的流和包含 "d" 的新流。 -
flatMap(Set::stream) (方法一) / flatMap(Function.identity()) (方法二): flatMap 是关键。它将Stream中的每个元素(在这里是一个Set
或Stream )转换为一个流,然后将所有这些流合并成一个单独的流。 Set::stream 将 Set 转换为 Stream。Function.identity() 则是将 Stream 自身作为流进行合并。 - collect(Collectors.toUnmodifiableSet()): 最后,collect 方法使用 Collectors.toUnmodifiableSet() 将流中的所有元素收集到一个新的不可变集合中。
注意事项:
- Collectors.toUnmodifiableSet() 是 Java 10 引入的,用于创建不可变集合。如果使用 Java 8 或 9,可以使用 Collectors.collectingAndThen(Collectors.toSet(), Collections::unmodifiableSet) 来代替,以达到相同的效果。
- 新集合中元素的顺序可能与原始集合不同,因为 Set 接口不保证元素的顺序。
- flatMap 是一个非常强大的操作,它可以用于处理各种嵌套数据结构,并将它们扁平化为单个流。
总结:
通过使用Stream API的flatMap操作和Collectors.toUnmodifiableSet()收集器,我们可以方便地从现有的不可变集合创建一个新的包含更多元素的不可变集合。 这种方法避免了直接修改原始集合,从而保持了其不可变性,并且避免了混合类型的集合。










