Сумма определенных значений HashMap в списке, только если есть совпадение значений под другим ключом - PullRequest
0 голосов
/ 10 апреля 2019

У меня есть ArrayList, заполненный тысячами HashMaps с четырьмя ключами и сопоставленными с ними значениями, построенными следующим образом:

HashMap<String, Object> map = new HashMap<>();
map.put("ID", 1);
map.put("NAME", name);
map.put("WORK_TIME", workTime);
map.put("ACCOUNT", name);

теперь с ArrayList из этих HashMaps. Я бы хотел суммировать время работы людей, имеющихтот же идентификатор, имя и учетная запись, например:

HashMap<String, Object> map1 = new HashMap<>();
map1.put("ID", 1);
map1.put("NAME", "Edward");
map1.put("WORK_TIME", 20);
map1.put("ACCOUNT", null);

HashMap<String, Object> map2 = new HashMap<>();
map2.put("ID", 1);
map2.put("NAME", "Krzych");
map2.put("WORK_TIME", 6);
map2.put("ACCOUNT", 123);

HashMap<String, Object> map3 = new HashMap<>();
map3.put("ID", 1);
map3.put("NAME", "Edward");
map3.put("WORK_TIME", 13.5);
map3.put("ACCOUNT", null);

HashMap<String, Object> map4 = new HashMap<>();
map4.put("ID", 2);
map4.put("NAME", "Grzesiek");
map4.put("WORK_TIME", 50);
map4.put("ACCOUNT", null);

HashMap<String, Object> map5 = new HashMap<>();
map5.put("ID", 2);
map5.put("NAME", "Edward");
map5.put("WORK_TIME", 12);
map5.put("ACCOUNT", 123);

[..]

ArrayList<HashMap<String,Object>> arrList = new ArrayList<>();
arrList.put..

, в результате я должен получить ArrayList из 4 HashMaps, где только два Edward HashMaps с одинаковыми идентификатором и учетной записью были объединены в одну с рабочим временем 33,5.

Единственное, что мне пришло в голову, - это итерировать по всем картам, сравнивать эти три значения и затем успешно заменять значения рабочего времени в хеш-картах, хранящихся во втором массиве. Я работаю в Java 8 и хочу выполнитьэто с помощью потоков, это возможно?Или вы видите лучшее решение?

1 Ответ

0 голосов
/ 10 апреля 2019

это лямбда:

Map<String, List<HashMap<String, Object>>> xxx = arrList.stream().collect(Collectors.groupingBy(x -> ((int) x.get("ID") + "") + x.get("NAME")));

дает результат, как показано ниже:

enter image description here

но, как упоминалось выше - вы должны использовать (если возможно) объекты вместо массивов. С объектами лямбда будет выглядеть так:

Map<String, List<Person>> xxx = arrList.stream().collect(Collectors.groupingBy(x -> x.getName() + x. getId()));

но, поскольку у вас есть класс Person, вы можете перезаписывать equals в соответствии со своими потребностями, и вы можете отображать и суммировать рабочее время в одной простой лямбде:

Map<Person, Double> result = arrList.stream().collect(Collectors.groupingBy(x -> x, Collectors.summingDouble(Person::getWorkTime)));

и это результат - карта Person в качестве ключа и сводка рабочего времени в качестве значения. enter image description here

код для класса Person:

public class Person {
        String name;
        int id;
        double workTime;

        public Person(String name, int id, double workTime) {
            this.name = name;
            this.id = id;
            this.workTime = workTime;
        }

        @Override
        public boolean equals(Object o) {
            if (this == o) return true;
            if (o == null || getClass() != o.getClass()) return false;
            Person person = (Person) o;
            return id == person.id &&
                    Objects.equals(name, person.name);
        }

        @Override
        public int hashCode() {

            return Objects.hash(name, id);
        }
    }

пример кода:

Person person1 = new Person("Edward", 1, 20);
Person person2 = new Person("Krzych", 1, 6);
Person person3 = new Person("Edward", 1, 13.5);
Person person4 = new Person("Grzesiek", 2, 50);
Person person5 = new Person("Edward", 2, 12);

ArrayList<Person> arrList = new ArrayList<>();

arrList.add(person1);
arrList.add(person2);
arrList.add(person3);
arrList.add(person4);
arrList.add(person5);

Map<Person, Double> result = arrList.stream().collect(Collectors.groupingBy(x -> x, Collectors.summingDouble(Person::getWorkTime)));
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...