首页 > 解决方案 > java中将JSON数组字符串转换为列表的通用方法

问题描述

我一直在尝试为我的项目创建一个实用程序,让您List<DesiredTypeClass>使用相同的 JSON 字符串进行制作。

以下是我目前的实现:

public static <T> List<?> stringToClassList(String data, Class<T> convertType, ObjectMapper mapper) {
    if (!StringUtils.isBlank(data)) {
        try {
            if (mapper.readTree(data).size() != 0) {
                return Arrays.asList(mapper.convertValue(mapper.readTree(data), Array.newInstance(convertType, 0).getClass()));
            }
        }
        catch (JsonProcessingException processingException) {
            LOGGER.error("Error coverting String to Json, Error: {}", processingException);
            throw new UnrecoverableRequestException(processingException);
        }
     }
     return new ArrayList<>();
}

要调用这个方法,你应该有:arrayString、一个classType(例如:使用MyDto.class 来获取List)和一个ObjectMapper 实例;

围绕这个编写一个测试用例,我发现结果中有额外的数组括号:

java.lang.AssertionError: 
Expected :[{"startDate":"2020-12-01","endDate":"2020-12-02"},{"startDate":"2020-12-10","endDate":"2020-12-11"}]
Actual   :[[{"startDate":"2020-12-01","endDate":"2020-12-02"},{"startDate":"2020-12-10","endDate":"2020-12-11"}]]

这是失败测试的简化版本:

@Test
public void testStringToClassList() {
    ObjectMapper objectMapper = new ObjectMapper();
    ObjectNode firstNode = new ObjectNode(JsonNodeFactory.instance);
    firstNode.set("startDate", JsonNodeFactory.instance.textNode("2020-12-01"));
    firstNode.set("endDate", JsonNodeFactory.instance.textNode("2020-12-02"));
    ObjectNode secondNode = new ObjectNode(JsonNodeFactory.instance);
    secondNode.set("startDate", JsonNodeFactory.instance.textNode("2020-12-10"));
    secondNode.set("endDate", JsonNodeFactory.instance.textNode("2020-12-11"));
    ArrayNode arrayNode = new ArrayNode(JsonNodeFactory.instance);
    arrayNode.add(firstNode);
    arrayNode.add(secondNode);
    String arrayString = "[{\"startDate\":\"2020-12-01\",\"endDate\":\"2020-12-02\"},{\"startDate\":\"2020-12-10\",\"endDate\":\"2020-12-11\"}]";
    List<?> objectList = ObjectNodeUtils.stringToClassList(arrayString, Object.class, objectMapper);
    Assert.assertEquals(arrayNode, objectMapper.valueToTree(objectList));
}

我理解这个问题与:

Arrays.asList(mapper.convertValue(mapper.readTree(data), Array.newInstance(convertType, 0).getClass()))

关于如何在此处获取列表的任何想法?(我不太习惯编写通用代码,但这似乎是一个不错的选择)

标签: javajsongenericsjackson

解决方案


不要重新发明轮子:

public static <T> List<T> stringToClassList(String data, Class<T> convertType, ObjectMapper mapper) {
    return mapper.readValue(data, mapper.getTypeFactory().constructCollectionType(List.class, convertType));
}

另请注意将返回类型从List<?>(不太有用)固定为List<T>(有用)。


您的测试方法也可以简化和改进:

@Test
public void testStringToClassList() {
    String input = "[\"foo\", \"bar\"]";
    List<String> expected = Arrays.asList("foo", "bar");
    Assert.assertEquals(expected, stringToClassList(input, String.class));
}

推荐阅读