Как использовать Джексона для десериализации массива объектов

534

Документация по связыванию данных Jackson указывает, что Jackson поддерживает десериализацию "Массивы всех поддерживаемых типов", но я не могу определить точный синтаксис для это.

Для одного объекта я бы сделал следующее:

//json input
{
    "id" : "junk",
    "stuff" : "things"
}

//Java
MyClass instance = objectMapper.readValue(json, MyClass.class);

Теперь для массива я хочу сделать это:

//json input
[{
    "id" : "junk",
    "stuff" : "things"
},
{
    "id" : "spam",
    "stuff" : "eggs"
}]

//Java
List<MyClass> entries = ?

Кто-нибудь знает, есть ли недостающая команда? Если нет, то каково решение?

  • 2
    Я предпочитаю библиотеку Google GSON для работы с JSON. Стоит проверить, если вы еще не пробовали ... делает работу с ней очень простой и интуитивно понятной.
  • 9
    FWIW Возможные решения этой конкретной проблемы с Gson практически идентичны возможным с помощью API привязки данных Джексона.
Показать ещё 1 комментарий
Теги:
jackson

6 ответов

1155
Лучший ответ

Сначала создайте mapper:

import com.fasterxml.jackson.databind.ObjectMapper;// in play 2.3
ObjectMapper mapper = new ObjectMapper();

Как массив:

MyClass[] myObjects = mapper.readValue(json, MyClass[].class);

As List:

List<MyClass> myObjects = mapper.readValue(jsonInput, new TypeReference<List<MyClass>>(){});

Другой способ указать тип списка:

List<MyClass> myObjects = mapper.readValue(jsonInput, mapper.getTypeFactory().constructCollectionType(List.class, MyClass.class));
  • 31
    Еще одно замечание: если при разборе вы получаете ошибку, такую как JsonMappingException: No suitable constructor found for type это означает, что вам нужно добавить конструктор по умолчанию в ваш класс, добавив частный конструктор без аргументов, исправив его для меня.
  • 11
    @SyntaxRules добавление явного конструктора необходимо, если у вас есть явный конструктор - если нет, компилятор автоматически создает открытый «пустой» конструктор. Хорошая точка зрения. Другая распространенная проблема заключается в том, что внутренние классы должны быть static иначе у них никогда не будет конструктора с нулевым аргументом.
Показать ещё 14 комментариев
137

Из Евгений Цховребов

List<MyClass> myObjects = Arrays.asList(mapper.readValue(json, MyClass[].class))

Это решение кажется лучшим для меня

  • 0
    Для тех, кто работает с агентами в Java, Lotus Domino, это путь. Я пробовал некоторые другие решения, но всегда получал ResourceNotFoundException
  • 0
    Для этого решения может потребоваться добавление SyntaxRules в комментариях к ответу выше, как и мы, именно для меня. Я просто хотел добавить это, чтобы оно не потерялось.
Показать ещё 7 комментариев
16

Для общей реализации:

public static <T> List<T> parseJsonArray(String json,
                                         Class<T> classOnWhichArrayIsDefined) 
                                         throws IOException, ClassNotFoundException {
   ObjectMapper mapper = new ObjectMapper();
   Class<T[]> arrayClass = (Class<T[]>) Class.forName("[L" + classOnWhichArrayIsDefined.getName() + ";");
   T[] objects = mapper.readValue(json, arrayClass);
   return Arrays.asList(objects);
}
  • 2
    Хорошая конструкция класса <T []>. Никогда этого не видел. Где вы нашли информацию об этом?
5

Сначала создайте экземпляр ObjectReader, который является потокобезопасным.

ObjectMapper objectMapper = new ObjectMapper();
ObjectReader objectReader = objectMapper.reader().forType(new TypeReference<List<MyClass>>(){});

Затем используйте его:

List<MyClass> result = objectReader.readValue(inputStream);
5
try {
    ObjectMapper mapper = new ObjectMapper();
    JsonFactory f = new JsonFactory();
    List<User> lstUser = null;
    JsonParser jp = f.createJsonParser(new File("C:\\maven\\user.json"));
    TypeReference<List<User>> tRef = new TypeReference<List<User>>() {};
    lstUser = mapper.readValue(jp, tRef);
    for (User user : lstUser) {
        System.out.println(user.toString());
    }

} catch (JsonGenerationException e) {
    e.printStackTrace();
} catch (JsonMappingException e) {
    e.printStackTrace();
} catch (IOException e) {
    e.printStackTrace();
}
0

вот утилита, которая преобразует json2object или Object2json, независимо от вашего pojo (entity T)

import java.io.IOException;
import java.io.StringWriter;
import java.util.List;

import com.fasterxml.jackson.core.JsonGenerationException;
import com.fasterxml.jackson.core.JsonParseException;
import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.JsonMappingException;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializationFeature;

/**
 * 
 * @author TIAGO.MEDICI
 * 
 */
public class JsonUtils {

    public static boolean isJSONValid(String jsonInString) {
        try {
            final ObjectMapper mapper = new ObjectMapper();
            mapper.readTree(jsonInString);
            return true;
        } catch (IOException e) {
            return false;
        }
    }

    public static String serializeAsJsonString(Object object) throws JsonGenerationException, JsonMappingException, IOException {
        ObjectMapper objMapper = new ObjectMapper();
        objMapper.enable(SerializationFeature.INDENT_OUTPUT);
        objMapper.disable(SerializationFeature.FAIL_ON_EMPTY_BEANS);
        StringWriter sw = new StringWriter();
        objMapper.writeValue(sw, object);
        return sw.toString();
    }

    public static String serializeAsJsonString(Object object, boolean indent) throws JsonGenerationException, JsonMappingException, IOException {
        ObjectMapper objMapper = new ObjectMapper();
        if (indent == true) {
            objMapper.enable(SerializationFeature.INDENT_OUTPUT);
            objMapper.disable(SerializationFeature.FAIL_ON_EMPTY_BEANS);
        }

        StringWriter stringWriter = new StringWriter();
        objMapper.writeValue(stringWriter, object);
        return stringWriter.toString();
    }

    public static <T> T jsonStringToObject(String content, Class<T> clazz) throws JsonParseException, JsonMappingException, IOException {
        T obj = null;
        ObjectMapper objMapper = new ObjectMapper();
        obj = objMapper.readValue(content, clazz);
        return obj;
    }

    @SuppressWarnings("rawtypes")
    public static <T> T jsonStringToObjectArray(String content) throws JsonParseException, JsonMappingException, IOException {
        T obj = null;
        ObjectMapper mapper = new ObjectMapper();
        obj = mapper.readValue(content, new TypeReference<List>() {
        });
        return obj;
    }

    public static <T> T jsonStringToObjectArray(String content, Class<T> clazz) throws JsonParseException, JsonMappingException, IOException {
        T obj = null;
        ObjectMapper mapper = new ObjectMapper();
        mapper = new ObjectMapper().configure(DeserializationFeature.ACCEPT_SINGLE_VALUE_AS_ARRAY, true);
        obj = mapper.readValue(content, mapper.getTypeFactory().constructCollectionType(List.class, clazz));
        return obj;
    }

Ещё вопросы

Сообщество Overcoder
Наверх
Меню