web-dev-qa-db-fra.com

Java 8 List <Map <String, Object >> to List <Map <String, Object >> grouper par clé et compter par valeur

J'ai la liste de cartes suivante

List<Map<String, Object>> listBeforeGroup = new ArrayList<Map<String, Object>>();

    Map<String, Object> m1 = new HashMap<String, Object>();
    m1.put("company", "LG");
    m1.put("billType", "A");
    m1.put("billPeriod", "09-2018");

    Map<String, Object> m2 = new HashMap<String, Object>();
    m2.put("company", "LG");
    m2.put("billType", "A");
    m2.put("billPeriod", "09-2018");

    Map<String, Object> m3 = new HashMap<String, Object>();
    m3.put("company", "LG");
    m3.put("billType", "A");
    m3.put("billPeriod", "09-2018");

    Map<String, Object> m4 = new HashMap<String, Object>();
    m4.put("company", "LG");
    m4.put("billType", "B");
    m4.put("billPeriod", "01-2019");

    Map<String, Object> m5 = new HashMap<String, Object>();
    m5.put("company", "LG");
    m5.put("billType", "B");
    m5.put("billPeriod", "01-2019");

    Map<String, Object> m6 = new HashMap<String, Object>();
    m6.put("company", "Samsung");
    m6.put("billType", "A");
    m6.put("billPeriod", "10-2018");

    Map<String, Object> m7 = new HashMap<String, Object>();
    m7.put("company", "Samsung");
    m7.put("billType", "A");
    m7.put("billPeriod", "10-2018");

    Map<String, Object> m8 = new HashMap<String, Object>();
    m8.put("company", "Samsung");
    m8.put("billType", "B");
    m8.put("billPeriod", "11-2018");

    listBeforeGroup.add(m1);listBeforeGroup.add(m2);
    listBeforeGroup.add(m3);listBeforeGroup.add(m4);
    listBeforeGroup.add(m5);listBeforeGroup.add(m6);

Comment obtenir cette sortie?

    //Desired Output - List<Map<String, Object>>
    //{company=LG, billType=A, billPeriod=09-2018, count=3}
    //{company=LG, billType=B, billPeriod=01-2019, count=2}
    //{company=Samsung, billType=A, billPeriod=10-2018, count=2}
    //{company=Samsung, billType=B, billPeriod=11-2018, count=1}

J'ai essayé cela, en utilisant Java 8 flux, mais je ne peux pas obtenir la sortie souhaitée

List<Map<String, Object>> listAfterGroup = listBeforeGroup.stream().map(m -> m.entrySet().stream().collect(Collectors.toMap(p -> p.getKey(), p - >  p.getValue()))).collect(Collectors.toList());

Et j'ai essayé ça, cette solution donne une carte mais je ne veux pas ça

Map<Object, Long> listAfterGroup = listBeforeGroup.stream().flatMap(m -> m.entrySet().stream()).collect(Collectors.groupingBy(Map.Entry::getKey,Collectors.counting()));

Je veux regrouper les cartes par la clé "billPeriod" par exemple, et compter les éléments par les valeurs, puis générer une nouvelle liste de cartes.

7
jhuamanchumo

Mon approche avec Java 8:

    Function<Map<String, Object>, String> createHashKey = map ->
                     map.values().stream()
                    .map(val -> String.valueOf(val))
                    .collect(Collectors.joining());

    BinaryOperator<Map<String, Object>> mergeDuplicate = (map1, map2) -> {
        int incrementedCount = (int)map1.get("count") + 1;
        map1.put("count", incrementedCount);
        return map1;
    };

    List<Map<String, Object>> listAfterGroup = listBeforeGroup.stream()
            .collect(Collectors.toMap(createHashKey, el -> {
                        el.put("count", 1);
                        return el;
            },mergeDuplicate))
          .values().stream()
          .collect(toList());

Ce n'est peut-être pas la solution la plus succincte, mais je pense qu'elle est assez lisible et facile à suivre la logique du code.

0
mwdev