给出以下 .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:无法反序列化
的实例 com.example.GeoPoint 超出 START_ARRAY 令牌 [来源:android.content.res.AssetManager$AssetInputStream@416a5850;行:5,列:25]
(通过参考链:com.example.Location["center"])
您的 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
}
}
]
我将此问题排序为验证来自 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
如前所述,Jackson 对象映射器会抛出
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;
}
另一方面,如果你没有像
Location
这样的pojo,你可以使用:
TypeReference<List<Map<String, Object>>> typeReference = new TypeReference<>() {};
return objectMapper.readValue(inputStream, typeReference);
应用上述解决方案后,我得到了积极的结果。 您的 JSON 字符串格式错误:center 的类型是无效对象的数组。将 JSON 字符串中经度和纬度周围的 [ 和 ] 替换为 { 和 },以便它们成为对象。 在 Angular pojo 类中,移动到 Java 端对象以存储为 POST 请求 - 会出现此类错误。 解决方案:将[和]替换为{和}即可解决问题。谢谢。
例如:
数据类
data class ToDo(
var id: Int,
var text: String?,
var completed: Boolean?) {}
当您使用 ToDo::class.java 但不使用 Array::class.java 作为 json 列表时,会引发上述反序列化错误
不起作用
private val mapper = ObjectMapper().registerKotlinModule()
.....
val todo= mapper.readValue(response.body(), ToDo::class.java)
com.fasterxml.jackson.databind.exc.MismatchedInputException:不能 从数组值反序列化
类型的值 (代币com.example.api.dto.ToDo
)位于[来源: (字符串)"[{"id":14,"text":"TEST","已完成":true},{"id":13,"text":"TEST","completed":true},{ "id":19,"text":"测试","已完成":true"[已截断 84 个字符];行:1,列:1]JsonToken.START_ARRAY
工作
private val mapper = ObjectMapper().registerKotlinModule()
.....
val todos = mapper.readValue(response.body(), Array<ToDo>::class.java)