如果必须一次性完成,则可以编写一个自定义收集器,将流缩减为最大元素列表。以下是基于
this answer
斯图尔特·马克斯。
List<MyObject> maxList = list.stream()
.collect(maxList(Comparator.comparing(MyObject::getSome)));
static <T> Collector<T,?,List<T>> maxList(Comparator<? super T> comp) {
return Collector.of(
ArrayList::new,
(list, t) -> {
int c;
if (list.isEmpty() || (c = comp.compare(t, list.get(0))) == 0) {
list.add(t);
} else if (c > 0) {
list.clear();
list.add(t);
}
},
(list1, list2) -> {
if (list1.isEmpty()) {
return list2;
}
if (list2.isEmpty()) {
return list1;
}
int r = comp.compare(list1.get(0), list2.get(0));
if (r < 0) {
return list2;
} else if (r > 0) {
return list1;
} else {
list1.addAll(list2);
return list1;
}
});
}
收集器将维护一个用于结果的ArrayList,并将每个元素累积到其中,检查该元素与当前列表的第一个元素的比较情况。零件
c = comp.compare(t, list.get(0))) == 0
将检查元素是否具有相同的最大值,如果是,则将其添加到列表中。