JsonMappingException:START_ARRAYトークンが不足しています


112

次の.jsonファイルがあるとします。

[
    {
        "name" : "New York",
        "number" : "732921",
        "center" : [
                "latitude" : 38.895111, 
                "longitude" : -77.036667
            ]
    },
    {
        "name" : "San Francisco",
        "number" : "298732",
        "center" : [
                "latitude" : 37.783333, 
                "longitude" : -122.416667
            ]
    }
]

含まれるデータを表すために2つのクラスを用意しました。

public class Location {
    public String name;
    public int number;
    public GeoPoint center;
}

...

public class GeoPoint {
    public double latitude;
    public double longitude;
}

.jsonファイルのコンテンツを解析するために、Jackson 2.2.xを使用して、次のメソッドを準備しました。

public static List<Location> getLocations(InputStream inputStream) {
    ObjectMapper objectMapper = new ObjectMapper();
    try {
        TypeFactory typeFactory = objectMapper.getTypeFactory();
        CollectionType collectionType = typeFactory.constructCollectionType(
                                            List.class, Location.class);
        return objectMapper.readValue(inputStream, collectionType);
    } catch (IOException e) {
        e.printStackTrace();
    }
    return null;
}

centerプロパティを省略している限り、すべてのコンテンツを解析できます。しかし、地理座標を解析しようとすると、次のエラーメッセージが表示されます。

com.fasterxml.jackson.databind.JsonMappingException:
[ソース:android.content.res.AssetManager$AssetInputStream@416a5850;にあるSTART_ARRAYトークンからcom.example.GeoPointのインスタンスを逆シリアル化できません 行:5、列:25]
(参照チェーンを通じて:com.example.Location ["center"])


jsonパーサーで簡単に解析できるのに、なぜjacksonを使用しているのですか
Shashank Agarwal 2013年

4
JSON文字列の形式が正しくcenterありません。タイプは無効なオブジェクトの配列です。交換してみ[]して{}周りのJSON文字列にlongitudeしてlatitude、彼らがオブジェクトになりますので。
Katona 2013年

1
@カトナありがとうございます。コメントを回答に変換して、質問を閉じることができますか?!
JJD 2013年

回答:


75

JSON文字列の形式が正しくcenterありません:のタイプは無効なオブジェクトの配列です。交換する[]して{}周りのJSON文字列にlongitudeしてlatitude、彼らがオブジェクトになりますので。

[
    {
        "name" : "New York",
        "number" : "732921",
        "center" : {
                "latitude" : 38.895111, 
                "longitude" : -77.036667
            }
    },
    {
        "name" : "San Francisco",
        "number" : "298732",
        "center" : {
                "latitude" : 37.783333, 
                "longitude" : -122.416667
            }
    }
]

1
私が抱えていたのと同じ問題ですが、Jackson 2.6.3vではまったく異なるクレイジーなエラーが発生し、2.9.8vではすべてが機能して以来、それはクレイジーです。くそー、これらのシリアル化エラーはとても迷惑です。
brebDev

114

JsonMappingException: out of START_ARRAY tokenそれは期待していますように例外はジャクソンオブジェクトマッパーによってスローされObject {}、それが見つかったのに対し、Array [{}]応答。

これは交換することによって解決することができるObjectObject[]の引数にgeForObject("url",Object[].class)。参照:

  1. Ref.1
  2. Ref.2
  3. Ref.3

9

私はこの問題をJSONLint.comからのjsonを検証し、それを修正するように並べ替えました。そして、これは同じためのコードです。

String jsonStr = "[{\r\n" + "\"name\":\"New York\",\r\n" + "\"number\": \"732921\",\r\n"+ "\"center\": {\r\n" + "\"latitude\": 38.895111,\r\n"  + " \"longitude\": -77.036667\r\n" + "}\r\n" + "},\r\n" + " {\r\n"+ "\"name\": \"San Francisco\",\r\n" +\"number\":\"298732\",\r\n"+ "\"center\": {\r\n" + "    \"latitude\": 37.783333,\r\n"+ "\"longitude\": -122.416667\r\n" + "}\r\n" + "}\r\n" + "]";

ObjectMapper mapper = new ObjectMapper();
MyPojo[] jsonObj = mapper.readValue(jsonStr, MyPojo[].class);

for (MyPojo itr : jsonObj) {
    System.out.println("Val of name is: " + itr.getName());
    System.out.println("Val of number is: " + itr.getNumber());
    System.out.println("Val of latitude is: " + 
        itr.getCenter().getLatitude());
    System.out.println("Val of longitude is: " + 
        itr.getCenter().getLongitude() + "\n");
}

注:MyPojo[].classは、jsonプロパティのゲッターとセッターを持つクラスです。

結果:

Val of name is: New York
Val of number is: 732921
Val of latitude is: 38.895111
Val of longitude is: -77.036667
Val of name is: San Francisco
Val of number is: 298732
Val of latitude is: 37.783333
Val of longitude is: -122.416667

1
これにより、私が数日にわたって抱えていた問題が修正されました。他の答えは意味がありませんでした。同じエラーを抱えている他の人のために。JSONに複数のオブジェクトが含まれている場合、複数のオブジェクトを返す必要があることに注意してください。私はMyObjectを返していましたが、MyObject []を返すはず
でした

8

言ったJsonMappingException: out of START_ARRAY token、それは期待していますように例外がジャクソンオブジェクトマッパーによってスローされObject {}、それが見つかったのに対し、Array [{}]応答。

より簡単な解決策は、メソッドgetLocationsを次のように置き換えることです。

public static List<Location> getLocations(InputStream inputStream) {
    ObjectMapper objectMapper = new ObjectMapper();
    try {
        TypeReference<List<Location>> typeReference = new TypeReference<>() {};
        return objectMapper.readValue(inputStream, typeReference);
    } catch (IOException e) {
        e.printStackTrace();
    }
    return null;
}

一方、のようなpojoがない場合はLocation、次のように使用できます。

TypeReference<List<Map<String, Object>>> typeReference = new TypeReference<>() {};
return objectMapper.readValue(inputStream, typeReference);

1
この代替実装を共有していただきありがとうございます。CollectionTypevsの使用に利点または欠点があるTypeReferenceかどうかわかりますか?
JJD 2018

1
TypeReference構文は他のものよりはるかに短いです。確かではありませんが、場合によっては型の消去に関連する反証があるかもしれません...しかし、私の世界でTypeReferenceは使用と理解が簡単です。
freedev 2018
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.