Як використовувати Джексона для десеріалізації масиву об’єктів


779

Документація, що зв'язує дані Джексона, вказує на те, що 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
Я віддаю перевагу бібліотеці GSON Google для роботи з JSON. Варто перевірити, чи ви ще не пробували цього ... робить роботу з ним дуже простою та інтуїтивно зрозумілою.
Джессі Вебб

11
FWIW Можливі рішення цієї конкретної проблеми з Gson майже ідентичні тим, що можливо з API прив'язки даних Джексона.
Програміст Брюс

17
Gweebz - можливо, ви хочете пояснити, чому ви вважаєте, що GSON - кращий вибір (порівняно з Джексоном)?
StaxMan

Відповіді:


1654

Спочатку створіть картограф:

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

Як масив:

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

Список:

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

Ще один спосіб вказати тип списку:

List<MyClass> myObjects = mapper.readValue(jsonInput, mapper.getTypeFactory().constructCollectionType(List.class, MyClass.class));

43
Ще одна примітка: якщо під час розбору ви отримуєте помилку, наприклад, JsonMappingException: No suitable constructor found for typeтоді це означає, що вам потрібно додати до свого класу конструктор за замовчуванням, додавши приватний конструктор без аргументів, який виправив мені.
SyntaxRules

12
@SyntaxRules додавання явного конструктора необхідно, якщо у вас є явний конструктор - якщо ні, компілятор автоматично створює публічний "порожній" конструктор. Гарна думка. Ще одна поширена проблема полягає в тому, що внутрішні класи повинні бути static- інакше вони ніколи не мають конструктора нульових аргументів.
StaxMan

244
Btw, List<MyClass> myObjects = Arrays.asList(mapper.readValue(json, MyClass[].class))працює в 10 разів швидше, ніж TypeRefence.

5
Я шукаю версію загального типу.
Стефан

1
@EugeneTskhovrebov ваш шлях найчистіший для вбудовування. Інші вимагають кастингу або декларування. Я пропоную додати його як власну відповідь, щоб допомогти виділити її та дати відповідь.
Ерік Б

190

Від Євгенія Цхоребова

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

Це рішення, здається, найкраще для мене


Для тих, хто працює з агентами на Java, Lotus Domino, це шлях. Я спробував деякі інші рішення, але завжди отримувавResourceNotFoundException
Джон

Для цього рішення може знадобитися додаток SyntaxRules в коментарях до відповіді вище, як і ми, це було для мене. Я просто хотів додати це, щоб воно не було втрачено.
Роб

2
абоArrays.asList(Json.fromJson(json.get("fieldName"), MyClass[].class))
Аль-Мотафар

3
абоList<MyClass> myObjects = Arrays.asList(mapper.treeToValue(jsonNode.get("fieldName"), MyClass[].class))
Collin Krawll

@CollinKrawll що робить objectmapper.treetovalue?
Eswar

35

Для загального впровадження:

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

4
Гарна конструкція класу <T []>. Ніколи цього не бачив. Де ви знайшли інформацію про це?
Roeland Van Heddegem

11

Спочатку створіть екземпляр ObjectReader, який захищений потоком.

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

Потім використовуйте його:

List<MyClass> result = objectReader.readValue(inputStream);

Це саме те, що я шукаю, дякую
самостійно

ми отримуємо - com.fasterxml.jackson.databind.JsonMappingException: Не вдається десаріалізувати екземпляр java.util.ArrayList з маркера START_OBJECT на [Джерело: java.io.FileInputStream@33fec21; рядок: 1, стовпчик: 1]
Дінеш Кумар

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

3

ось утиліта, яка перетворює json2object або Object2json, незалежно від вашого pojo (сутність 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;
    }

0

Ви також можете створити клас, який розширює ArrayList:

public static class MyList extends ArrayList<Myclass> {}

а потім використовувати його так:

List<MyClass> list = objectMapper.readValue(json, MyList.class);
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.