如何使用 Java 8 个流将列表元素映射到它们的索引?

How to map elements of the list to their indices using Java 8 streams?

有了一个字符串列表,我需要构建一个对象列表,这些对象实际上是对 (string, its position in the list)。目前我有这样的代码使用 google 集合:

public Robots(List<String> names) {
    ImmutableList.Builder<Robot> builder = ImmutableList.builder();
    for (int i = 0; i < names.size(); i++) {
        builder.add(new Robot(i, names.get(i)));
    }
    this.list = builder.build();
}

我想使用 Java 8 个流来执行此操作。如果没有索引,我可以这样做:

public Robots(List<String> names) {
    this.list = names.stream()
            .map(Robot::new) // no index here
            .collect(collectingAndThen(
                    Collectors.toList(),
                    Collections::unmodifiableList
            ));
}

要获取索引,我必须这样做:

public Robots(List<String> names) {
    AtomicInteger integer = new AtomicInteger(0);
    this.list = names.stream()
            .map(string -> new Robot(integer.getAndIncrement(), string))
            .collect(collectingAndThen(
                    Collectors.toList(),
                    Collections::unmodifiableList
            ));
}

然而,文档说映射函数应该是无状态的,但 AtomicInteger 实际上是它的状态。

有没有办法将顺序流的元素映射到它们在流中的位置?

你可以这样做:

public Robots(List<String> names) {
    this.list = IntStream.range(0, names.size())
                         .mapToObj(i -> new Robot(i, names.get(i)))
                         .collect(collectingAndThen(toList(), Collections::unmodifiableList));
}

但是,根据列表的底层实现,它可能效率不高。您可以从 IntStream; 中获取一个迭代器;然后在 mapToObj.

中调用 next()

作为替代方案,proton-pack 库定义了流的 zipWithIndex 功能:

 this.list = StreamUtils.zipWithIndex(names.stream())
                        .map(i -> new Robot(i.getIndex(), i.getValue()))
                        .collect(collectingAndThen(toList(), Collections::unmodifiableList));

最简单的方法是流索引:

List<Robot> robots = IntStream.range(0, names.size())
                              .mapToObj(i -> new Robot(i, names.get(i))
                              .collect(toList());