java – 在基本流上使用收集器
作者:互联网
在Java 8中是否有任何方式在原始流上使用Stream::collect(Collector)?
通常是流<整数>例如有两种收集方法:
> <R,A> R collect(Collector<? super T,A,R> collector)
> <R> R collect(Supplier<R> supplier,
BiConsumer<R,? super T> accumulator,
BiConsumer<R,R> combiner)
但IntStream只有一种收集方法:
> <R> R collect(Supplier<R> supplier,
ObjIntConsumer<R> accumulator,
BiConsumer<R,R> combiner)
现在作为示例代码,我有以下内容:
@Override
public void run() {
result = LongStream.range(1, maximum).boxed()
.collect(Collectors.toMap(i -> i, i -> (int)Iterators.longStream(new CollatzGenerator(i)).count()))
.entrySet().stream()
.max(Comparator.comparingLong(Map.Entry::getValue))
.get().getKey();
}
正如您所看到的,我首先打包基元以便能够使用收集器.方法.
有什么方法可以使用原语,并且仍然与Collectors.toMap具有相同的代码
?
解决方法:
由于Map是Generic接口,因此无法在没有装箱的情况下创建Map.但是,当你想要的只是创建另一个流(只有两个值包含在Map.Entry中)时,将项目收集到Map中是没有意义的.您只需创建Map.Entry实例而无需收集值:
LongStream.range(1, maximum)
.mapToObj(i->new AbstractMap.SimpleEntry<>(i, Iterators.longStream(new CollatzGenerator(i)).count()))
.max(Comparator.comparingLong(Map.Entry::getValue))
.get().getKey();
这仍然可以进行自动装箱,但是一旦你在这一点上,你可以通过自己创建一个合适的值持有者类来摆脱Map.Entry:
static final class TwoLongs {
final long key, value;
TwoLongs(long k, long v) { key=k; value=v; }
public long getKey() { return key; }
public long getValue() { return value; }
}
使用此持有者类,您可以处理您的数据而无需装箱:
LongStream.range(1, maximum)
.mapToObj(i->new TwoLongs(i, Iterators.longStream(new CollatzGenerator(i)).count()))
.max(Comparator.comparingLong(TwoLongs::getValue))
.get().getKey();
好吧,它仍然是某种拳击,但创建一个项目(TwoLongs实例)而不是三个(一个Map.Entry和两个Longs).
标签:java,java-8,java-stream,boxing,primitive 来源: https://codeday.me/bug/20190624/1279871.html