我想知道有没有别的办法
List<X> lastN = all.subList(Math.max(0, all.size() - n), all.size());
使用?
发布于 2015-05-27 17:00:00
自定义收集器可以这样编写:
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);
}
并像这样使用它:
List<String> lastTen = input.stream().collect(lastN(10));
发布于 2015-05-27 16:58:39
如果流的大小未知,那么可能没有办法消耗整个流并缓冲到目前为止遇到的最后一个n
元素。您可以使用某种类型的双端队列或专门的环形缓冲区自动保持其最大大小(有关某些实现,请参阅this related question )。
public static <T> List<T> lastN(Stream<T> stream, int n) {
Deque<T> result = new ArrayDeque<>(n);
stream.forEachOrdered(x -> {
if (result.size() == n) {
result.pop();
}
result.add(x);
});
return new ArrayList<>(result);
}
所有这些操作(size
,pop
,add
)的复杂度都应该为O(1),因此长度为(未知)n的流的总体复杂度将为O(n)。
发布于 2019-08-08 17:57:19
有时我需要一个"oneliner“(在本例中是一个三行代码),因为创建一个收集器太麻烦了。
如果流很小,那么可以在不牺牲太多性能的情况下再次执行reverse
、limit
和reverse
。这将产生最后n个元素。
如果需要过滤,则它很有用,因为在这种情况下,无法指定大小。
Stream.of(1, 2, 3, 4, 5, 6, 7, 8, 9)
.filter(i -> i % 2 == 0)
.sorted(Comparator.reverseOrder())
.limit(2)
.sorted(Comparator.naturalOrder())
.forEach(System.out::println); // prints 6 8
https://stackoverflow.com/questions/30476127
复制相似问题