将Java类转换为List <list <string >>以显示在表中

时间:2018-09-09 22:22:06

标签: java list lambda hashmap set

我想在Java中的某些类之间进行转换,以显示在表中。 我收到一个List<Version>(删除了getter / setter使其变得更短):

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项(DEV的3项和UK的4项),列表中的每个项目(abc)中恰好有3个工作,最后,对于a,我将在所有7个条目中使用1.1.1版工作a,而对于b,我将在DEV中拥有所有3个10.0.1版本,但在服务器uk-4.lan中拥有3个10.0.0版本和1个10.0.1版本。英国。 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)", ""),
        )
    );

但是,如果您向我展示如何使用另一种类型的结构生成表,那也是有效的。

非常感谢, 乔奥

1 个答案:

答案 0 :(得分:0)

恐怕围绕“哪个服务器不相同”问题您的问题尚不完全清楚。我认为定义不明确。例如,如果您有四台服务器,每台服务器具有不同的作业版本?

考虑到这一点,我假设您想要通过以下测试:

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 lambda的有趣练习。我找到的解决方案是:

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);
        }
    }
}