147

给定以下 .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:无法
从 [Source: android.content.res.AssetManager$AssetInputStream@416a5850; 的 START_ARRAY 令牌中反序列化 com.example.GeoPoint 的实例;行:5,列:25]
(通过引用链:com.example.Location["center"])

4

4 回答 4

157

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

这可以通过在参数中替换ObjectObject[]来解决geForObject("url",Object[].class)。参考:

  1. 参考文献1
  2. 参考文献2
  3. 参考文献 3
于 2015-11-04T07:14:06.683 回答
89

您的 JSON 字符串格式错误:类型center是无效对象的数组。在 JSON 字符串中用and替换[and ,因此它们将成为对象:]{}longitudelatitude

[
    {
        "name" : "New York",
        "number" : "732921",
        "center" : {
                "latitude" : 38.895111, 
                "longitude" : -77.036667
            }
    },
    {
        "name" : "San Francisco",
        "number" : "298732",
        "center" : {
                "latitude" : 37.783333, 
                "longitude" : -122.416667
            }
    }
]
于 2013-10-16T18:25:38.623 回答
18

我将此问题分类为验证来自 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
于 2018-12-27T06:44:27.277 回答
13

如前所述,JsonMappingException: out of START_ARRAY tokenJackson 对象映射器会抛出异常,因为它期待一个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 like Location,你可以使用:

TypeReference<List<Map<String, Object>>> typeReference = new TypeReference<>() {};
return objectMapper.readValue(inputStream, typeReference);
于 2018-02-02T15:52:24.887 回答