Преобразование класса Java в список <List <String>> для отображения в таблице - PullRequest
0 голосов
/ 10 сентября 2018

Я хотел бы сделать преобразование между некоторыми классами в Java для отображения в таблице. Я получаю List<Version> (удален метод получения / установки, чтобы сделать его короче):

public class Version {
    private String server;
    private List<Job> jobs;
}
public class Job {
    private String name;
    private String version;
}

Теперь у меня есть инструмент для проверки данных серверов для всех заданий: - Каждый элемент списка представляет собой информацию об одном сервере, поэтому несколько серверов содержат несколько заданий, но у каждого из серверов одинаковые задания, но версия может отличаться. - Итак, Version.server - это имя сервера, Job.name - имя задания, а Job.version - версия задания. - Кроме того, все серверы запускаются с одинаковыми именами, например: dev-1.lan, dev-2.lan, uk-1.lan, us-1.lan и т. Д .;

Благодаря @Korolar, вот правильный ввод:

   var input = List.of(
            new Version("dev-1.lan", List.of(
                new Job("a", "1.1.1"),
                new Job("b", "10.0.1"),
                new Job("c", "2.0.1")
            )),
            new Version("dev-2.lan", List.of(
                new Job("a", "1.1.1"),
                new Job("b", "10.0.1"),
                new Job("c", "2.0.1")
            )),
            new Version("dev-3.lan", List.of(
                new Job("a", "1.1.1"),
                new Job("b", "10.0.1"),
                new Job("c", "2.0.2")
            )),
            new Version("uk-1.lan", List.of(
                new Job("a", "1.1.1"),
                new Job("b", "10.0.0"),
                new Job("c", "2.0.2")
            )),
            new Version("uk-2.lan", List.of(
                new Job("a", "1.1.1"),
                new Job("b", "10.0.0"),
                new Job("c", "2.0.2")
            )),
            new Version("uk-3.lan", List.of(
                new Job("a", "1.1.1"),
                new Job("b", "10.0.0"),
                new Job("c", "2.0.2")
            )),
            new Version("uk-4.lan", List.of(
                new Job("a", "1.1.1"),
                new Job("b", "10.0.1"),
                new Job("c", "2.0.2")
            ))
        );

Я хочу преобразовать его в таблицу, чтобы ее было легче читать. Где я хочу объединить задания с одинаковыми именем и версией и, если версия не совпадает, добавить еще одну строку и указать, какой сервер не с той же версией, например:

+------------+--------------+------------+
| job        | DEV          | UK         |
+------------+--------------+------------+
| a          | 1.1.1        | 1.1.1      |
| b          | 10.0.1       | 10.0.0     |
| b          |              | 10.0.1 (4) |
| c          | 2.0.1        | 2.0.2      |
| c          | 2.0.1 (3)    |            |
+------------+--------------+------------+

Хорошо, так что, как вы можете видеть, (x) является сервером, который не прав, например, я получил List<Version> по крайней мере 7 предметов (3 от DEV и 4 от UK), с ровно 3 задания в каждом элементе списка (a, b и c) и, наконец, для a у меня будет версия 1.1.1 во всех 7 записях для задания a, а для b Я бы имел все 3 версии 10.0.1 в DEV, но 3 версии 10.0.0 и 1 версию 10.0.1, которая находится на сервере uk-4.lan в Великобритании. Аналогичные результаты будут иметь место и для c, но сервер с другими результатами будет сервером dev-3.lan.

Теперь мой вопрос: как я могу преобразовать List<Version> в простой List<List<String>> для представления этой таблицы или чего-то подобного? Не беспокойтесь о печати, потому что я уже сделал эту часть.

Итак, я ожидаю, что мой результат будет примерно таким:

var expectedOutput = List.Of
        List.of("apps","DEV", "Uk"),
        List.of("a", "1.1.1", "1.1.1"),
        List.of("b", "10.0.1", "10.0.0"),
        List.of("b", "", "10.0.1 (4)"),
        List.of("c", "2.0.1", "2.0.2"),
        List.of("c", "2.0.2 (3)", ""),
        )
    );

Но если вы покажете мне, как я могу сгенерировать таблицу, используя структуру другого типа, это также допустимо.

Большое спасибо, Joao

1 Ответ

0 голосов
/ 10 сентября 2018

Боюсь, ваш вопрос не совсем понятен по поводу вопроса «какой сервер не тот». Я не думаю, что это хорошо определено. Например, что если у вас есть четыре сервера, каждый с разной версией задания?

Имея это в виду, я предположил, что вы хотели, чтобы следующий тест прошел:

import static org.junit.jupiter.api.Assertions.assertEquals;

import java.util.List;
import java.util.Map;
import java.util.Set;
import org.junit.jupiter.api.Test;

class QuestionTest {
    @Test
    void example() {
        var input = List.of(
            new Version("dev-1.lan", List.of(
                new Job("a", "1.1.1"),
                new Job("b", "10.0.1"),
                new Job("c", "2.0.1")
            )),
            new Version("dev-2.lan", List.of(
                new Job("a", "1.1.1"),
                new Job("b", "10.0.1"),
                new Job("c", "2.0.1")
            )),
            new Version("dev-3.lan", List.of(
                new Job("a", "1.1.1"),
                new Job("b", "10.0.1"),
                new Job("c", "2.0.2")
            )),
            new Version("uk-1.lan", List.of(
                new Job("a", "1.1.1"),
                new Job("b", "10.0.0"),
                new Job("c", "2.0.2")
            )),
            new Version("uk-2.lan", List.of(
                new Job("a", "1.1.1"),
                new Job("b", "10.0.0"),
                new Job("c", "2.0.2")
            )),
            new Version("uk-3.lan", List.of(
                new Job("a", "1.1.1"),
                new Job("b", "10.0.0"),
                new Job("c", "2.0.2")
            )),
            new Version("uk-4.lan", List.of(
                new Job("a", "1.1.1"),
                new Job("b", "10.0.1"),
                new Job("c", "2.0.2")
            ))
        );

        var expectedOutput = Map.of(
            "a", Map.of(
                "DEV", Map.of(
                    "1.1.1", Set.of(1, 2, 3)
                ),
                "UK", Map.of(
                    "1.1.1", Set.of(1, 2, 3, 4)
                )
            ),
            "b", Map.of(
                "DEV", Map.of(
                    "10.0.1", Set.of(1, 2, 3)
                ),
                "UK", Map.of(
                    "10.0.0", Set.of(1, 2, 3),
                    "10.0.1", Set.of(4)
                )
            ),
            "c", Map.of(
                "DEV", Map.of(
                    "2.0.1", Set.of(1, 2),
                    "2.0.2", Set.of(3)
                ),
                "UK", Map.of(
                    "2.0.2", Set.of(1, 2, 3, 4)
                )
            )
        );

        var actualOutput = Main.parse(input);

        assertEquals(expectedOutput, actualOutput);
    }
}

С этим предположением ваш вопрос сводится к забавному упражнению по использованию лямбд Java. Решение, которое я нашел:

import static java.util.stream.Collectors.groupingBy;
import static java.util.stream.Collectors.mapping;
import static java.util.stream.Collectors.toSet;

import java.util.List;
import java.util.Map;
import java.util.Set;

class Main {
    static Map<String, Map<String, Map<String, Set<Integer>>>> parse(List<Version> input) {
        return input.stream()
            .flatMap(version -> version.jobs.stream().map(job -> new Entry(version, job)))
            .collect(
                groupingBy(
                    entry -> entry.jobName,
                    groupingBy(
                        entry -> entry.serverName,
                        groupingBy(
                            entry -> entry.jobVersion,
                            mapping(entry -> entry.serverNumber, toSet())
                        )
                    )
                )
            );
    }
}

с классом помощника Entry, определенным как:

class Entry {
    final String serverName;
    final int serverNumber;
    final String jobName;
    final String jobVersion;

    Entry(Version version, Job job) {
        this.serverName = version.serverName;
        this.serverNumber = version.serverNumber;
        this.jobName = job.name;
        this.jobVersion = job.version;
    }
}

и с:

class Job {
    final String name;
    final String version;

    Job(String name, String version) {
        this.name = name;
        this.version = version;
    }
}

и наконец:

import java.util.List;
import java.util.Locale;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

class Version {
    private static final Pattern SERVER_NAME_PATTERN = Pattern.compile("(\\w+)-(\\d+)\\.lan");

    final String server;
    final List<Job> jobs;

    final String serverName;
    final int serverNumber;

    Version(String server, List<Job> jobs) {
        this.server = server;
        this.jobs = List.copyOf(jobs);

        Matcher matcher = SERVER_NAME_PATTERN.matcher(server);
        if (matcher.matches()) {
            this.serverName = matcher.group(1).toUpperCase(Locale.US);
            this.serverNumber = Integer.parseInt(matcher.group(2));
        } else {
            throw new IllegalArgumentException("Invalid server: " + server);
        }
    }
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...