从流中获取最后n个元素

ytt*_*rrr 28 java java-8 java-stream

我想知道是否有替代方案

List<X> lastN = all.subList(Math.max(0, all.size() - n), all.size());
Run Code Online (Sandbox Code Playgroud)

Tag*_*eev 22

自定义收集器可以这样写:

public static <T> Collector<T, ?, List<T>> lastN(int n) {
    return Collector.<T, Deque<T>, List<T>>of(ArrayDeque::new, (acc, t) -> {
        if(acc.size() == n)
            acc.pollFirst();
        acc.add(t);
    }, (acc1, acc2) -> {
        while(acc2.size() < n && !acc1.isEmpty()) {
            acc2.addFirst(acc1.pollLast());
        }
        return acc2;
    }, ArrayList::new);
}
Run Code Online (Sandbox Code Playgroud)

并像这样使用它:

List<String> lastTen = input.stream().collect(lastN(10));
Run Code Online (Sandbox Code Playgroud)

  • 你不需要编写`ArrayList <T> :: new`,只需`ArrayList :: new`就足够了,因为方法引用总是使用类型推断而不是原始类型(比如使用"钻石运算符"时总是存在) `:: new`).你已经将它与`ArrayDeque :: new`一起使用了.顺便说一句.我更喜欢`removeFirst` /`removeLast`而不是`pollFirst` /`pollLast`这里...... (2认同)
  • @Holger,我首先编写了`ArrayList :: new`,但Eclipse显示了一个关于未经检查的构造函数的警告.好吧,这可能是ECJ特有的问题. (2认同)

Jor*_*lla 19

使用Stream.skip()

在丢弃流的前n个元素后,返回由此流的其余元素组成的流.如果此流包含少于n个元素,则将返回空流.

all.stream().skip(Math.max(0, all.size() - n)).forEach(doSomething);
Run Code Online (Sandbox Code Playgroud)

  • 如果`all.size()<n`,则抛出`IllegalArgumentException`.原始问题中的`Math.max(0,all.size() - n)`也是必要的. (3认同)
  • 如果我们不知道流大小怎么办(假设最坏的情况)? (2认同)