JsonMappingException:超出START_ARRAY令牌


111

给定以下.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
            ]
    }
]

我准备了两个类来表示所包含的数据:

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:无法
从[START_ARRAY]令牌中反序列化com.example.GeoPoint 实例,位于[来源:android.content.res.AssetManager$AssetInputStream@416a5850; 行:5,列:25]
(通过参考链:com.example.Location [“ center”])


如果您可以简单地使用json解析器解析它,为什么还要使用jackson
Shashank Agarwal

4
您的JSON字符串格式错误,类型center为无效对象的数组。试图取代[,并]{}在周围的JSON字符串longitude,并latitude让他们将对象。
卡托纳2013年

1
@Katona谢谢。您能否将您的评论转换为答案,以便我结束问题?!
JJD 2013年

Answers:


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
我遇到了同样的问题,但是很疯狂,因为在杰克逊2.6.3v中存在一个完全不同的疯狂错误,而对于2.9.8v,一切正常。该死的,所以这些序列化错误太烦人了。
brebDev

113

JsonMappingException: out of START_ARRAY token杰克逊对象映射器抛出异常,因为它期望一个,Object {}而它却找到一个Array [{}]响应。

这可以通过更换得到解决ObjectObject[]在参数geForObject("url",Object[].class)。参考文献:

  1. 参考1
  2. 参考2
  3. 参考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属性的getter和setter方法的类。

结果:

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 []
Aaria

8

如前所述,JsonMappingException: out of START_ARRAY tokenJackson对象映射器会抛出异常,因为它期望an 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

1
TypeReference语法比其他语法短得多。不确定,在某些情况下是否可能存在与类型擦除有关的反指示...但是在我的世界TypeReference中,它更易于使用和理解。
freedev
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.