Java 8 не поддерживает порядок при группировке
Я использую Java 8 для группировки по данным. Но полученные результаты не упорядочены.
Map<GroupingKey, List<Object>> groupedResult = null;
if (!CollectionUtils.isEmpty(groupByColumns)) {
Map<String, Object> mapArr[] = new LinkedHashMap[mapList.size()];
if (!CollectionUtils.isEmpty(mapList)) {
int count = 0;
for (LinkedHashMap<String, Object> map : mapList) {
mapArr[count++] = map;
}
}
Stream<Map<String, Object>> people = Stream.of(mapArr);
groupedResult = people
.collect(Collectors.groupingBy(p -> new GroupingKey(p, groupByColumns), Collectors.mapping((Map<String, Object> p) -> p, toList())));
public static class GroupingKey
public GroupingKey(Map<String, Object> map, List<String> cols) {
keys = new ArrayList<>();
for (String col : cols) {
keys.add(map.get(col));
}
}
// Add appropriate isEqual() ... you IDE should generate this
@Override
public boolean equals(Object obj) {
if (obj == null) {
return false;
}
if (getClass() != obj.getClass()) {
return false;
}
final GroupingKey other = (GroupingKey) obj;
if (!Objects.equals(this.keys, other.keys)) {
return false;
}
return true;
}
@Override
public int hashCode() {
int hash = 7;
hash = 37 * hash + Objects.hashCode(this.keys);
return hash;
}
@Override
public String toString() {
return keys + "";
}
public ArrayList<Object> getKeys() {
return keys;
}
public void setKeys(ArrayList<Object> keys) {
this.keys = keys;
}
}
Здесь я использую свой класс groupingKey, с помощью которого я m динамически переходит из ux. Как получить эту группуByColumns в отсортированной форме?
Ответы
Ответ 1
Не поддерживается порядок - это свойство Map
, которое сохраняет результат. Если вам нужно определенное поведение Map
, вам нужно запросить конкретную реализацию Map
. Например. LinkedHashMap
поддерживает порядок вставки:
groupedResult = people.collect(Collectors.groupingBy(
p -> new GroupingKey(p, groupByColumns),
LinkedHashMap::new,
Collectors.mapping((Map<String, Object> p) -> p, toList())));
Кстати, нет причин копировать содержимое mapList
в массив до создания Stream
. Вы можете просто вызвать mapList.stream()
, чтобы получить соответствующий Stream
.
Кроме того, Collectors.mapping((Map<String, Object> p) -> p, toList())
является устаревшим. p->p
является тождественным отображением, поэтому нет причин запрашивать mapping
вообще:
groupedResult = mapList.stream().collect(Collectors.groupingBy(
p -> new GroupingKey(p, groupByColumns), LinkedHashMap::new, toList()));
Но даже GroupingKey
устарел. Он в основном обертывает значения List
, поэтому вы можете просто использовать ключ List
в первую очередь. List
выполнить hashCode
и equals
соответствующим образом (но вы не должны изменять этот ключ List
впоследствии).
Map<List<Object>, List<Object>> groupedResult=
mapList.stream().collect(Collectors.groupingBy(
p -> groupByColumns.stream().map(p::get).collect(toList()),
LinkedHashMap::new, toList()));
Ответ 2
На основании @Holger отличный ответ. Я публикую это, чтобы помочь тем, кто хочет сохранить порядок после группировки, а также изменить отображение.
Давайте упростим и предположим, что у нас есть список лиц (int age, String name, String addresss... и т.д.), И мы хотим, чтобы имена группировались по возрасту, сохраняя возраст по порядку:
final LinkedHashMap<Integer, List<String> map = myList
.stream()
.sorted(Comparator.comparing(p -> p.getAge())) //sort list by ages
.collect(Collectors.groupingBy(p -> p.getAge()),
LinkedHashMap::new, //keeps the order
Collectors.mapping(p -> p.getName(), //map name
Collectors.toList())));