Jacksonを使用してオブジェクトの配列を逆シリアル化する方法


780

ジャクソンデータバインディングドキュメントはジャクソンサポート「はサポートされているすべての型の配列を」deserialisingことを示しているが、私はこのために正確な構文を把握することはできません。

単一のオブジェクトの場合、これを行います。

//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
私は、JSONを処理するためにGoogleのGSONライブラリを好みます。まだ試していない場合は、チェックする価値があります...簡単かつ直感的に操作できます。
Jesse Webb

11
FWIW Gsonでのこの特定の問題に対する可能な解決策は、JacksonのデータバインディングAPIで可能な解決策とほとんど同じです。
プログラマBruce

17
グウィーブズ-GSONが(ジャクソンと比較して)良い選択だと思う理由を説明したいですか?
StaxMan

回答:


1657

最初にマッパーを作成します。

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
もう1つの注意点として、解析中にエラーが発生したJsonMappingException: No suitable constructor found for type場合は、デフォルトのコンストラクターをクラスに追加して、引数のないプライベートコンストラクターを追加する必要があることを意味します。
SyntaxRules 2013

12
明示的なコンストラクターがある場合、明示的なコンストラクターを追加する@SyntaxRulesが必要です。そうでない場合、コンパイラーは自動的にパブリックな「空の」コンストラクターを作成します。いい視点ね。もう1つの一般的な問題は、内部クラスが存在する必要があることstaticです。それ以外の場合、引数がゼロのコンストラクターはありません。
StaxMan 2013

244
ところで、List<MyClass> myObjects = Arrays.asList(mapper.readValue(json, MyClass[].class))TypeRefenceよりも最大10倍速く動作します。

5
ジェネリック型のバージョンを探しています。
Stephane

1
@EugeneTskhovrebovあなたの方法はインライン化のための最もきれいです。他のキャストまたは宣言が必要です。私はそれをそれを強調するのを助けるためにそれ自身の答えとして追加し、あなたにいくつかの担当者を与えることを勧めます。
エリックB

190

ユージーン・ツホブレボフから

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

この解決策は私にとって最高のようです


Java、Lotus Dominoでエージェントを使用している場合は、これがよい方法です。私は他の解決策のいくつかを試しましたが、常に得ましたResourceNotFoundException
John

上記の回答のコメントにSyntaxRulesを追加することは、このソリューションでは必要になる可能性があります。それが失われないように私はそれを追加したかっただけです。
Rob

2
またはArrays.asList(Json.fromJson(json.get("fieldName"), MyClass[].class))
Al-Mothafar 2017

3
またはList<MyClass> myObjects = Arrays.asList(mapper.treeToValue(jsonNode.get("fieldName"), MyClass[].class))
Collin Krawll 2018

@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
Class <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.io.FileInputStream@33fec21;のSTART_OBJECTトークンからjava.util.ArrayListのインスタンスをデシリアライズできません。行:1、列:1]
Dinesh Kumar

8
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

ここに、あなたのpojo(エンティティT)が何であれ、json2objectまたはObject2jsonを変換するユーティリティがあります。

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

0

私のリンターは未チェックのキャストを許可しないため、この回答を使用できませんでした。

ここにあなたが使用できる代替案があります。それは実際にはよりクリーンなソリューションだと思います。

public <T> List<T> parseJsonArray(String json, Class<T> clazz) throws JsonProcessingException {
  var tree = objectMapper.readTree(json);
  var list = new ArrayList<T>();
  for (JsonNode jsonNode : tree) {
    list.add(objectMapper.treeToValue(jsonNode, clazz));
  }
  return list;
}
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.