在这种情况下,只有奇数行具有有意义的数据,并且没有可以唯一标识这些行的字符。我的意图是得到等同于以下示例的内容:

Stream<DomainObject> res = Files.lines(src)
     .filter(line -> isOddLine())
     .map(line -> toDomainObject(line))

有没有共享全局状态的“干净”方法?

最佳答案

一种干净的方法是更深入地实现Spliterator。在此级别上,您可以控制流元素的迭代,并且只要下游请求一个项目,就可以简单地对两个项目进行迭代:

public class OddLines<T> extends Spliterators.AbstractSpliterator<T>
    implements Consumer<T> {

    public static <T> Stream<T> oddLines(Stream<T> source) {
        return StreamSupport.stream(new OddLines(source.spliterator()), false);
    }
    private static long odd(long l) { return l==Long.MAX_VALUE? l: (l+1)/2; }
    Spliterator<T> originalLines;

    OddLines(Spliterator<T> source) {
        super(odd(source.estimateSize()), source.characteristics());
        originalLines=source;
    }

    @Override
    public boolean tryAdvance(Consumer<? super T> action) {
        if(originalLines==null || !originalLines.tryAdvance(action))
            return false;
        if(!originalLines.tryAdvance(this)) originalLines=null;
        return true;
    }

    @Override
    public void accept(T t) {}
}

然后你可以像这样使用它
Stream<DomainObject> res = OddLines.oddLines(Files.lines(src))
    .map(line -> toDomainObject(line));

该解决方案没有副作用,并且保留了Stream API的大多数优点(如惰性评估)。但是,应该清楚的是,它对于无序流处理没有有用的语义(请注意诸如对所有元素执行终端操作时使用forEachOrdered而不是forEach的微妙方面),并且在原则上支持并行处理时,不太可能非常有效率...

10-01 15:19