我正在使用Jackson库来解析来自服务器的大型JSON响应。 json的大小约为7-8 mb。
我在这段代码上得到了outOfMemoryError:
ObjectMapper mapper = new ObjectMapper();
JsonNode rootParser = mapper.readValue(is, JsonNode.class);
这是我得到的例外:
01-14 13:13:20.103: E/AndroidRuntime(25468): FATAL EXCEPTION: Thread-13
01-14 13:13:20.103: E/AndroidRuntime(25468): java.lang.OutOfMemoryError
01-14 13:13:20.103: E/AndroidRuntime(25468): at java.util.ArrayList.add(ArrayList.java:123)
01-14 13:13:20.103: E/AndroidRuntime(25468): at org.codehaus.jackson.node.ArrayNode._add(ArrayNode.java:722)
01-14 13:13:20.103: E/AndroidRuntime(25468): at org.codehaus.jackson.node.ArrayNode.add(ArrayNode.java:203)
01-14 13:13:20.103: E/AndroidRuntime(25468): at org.codehaus.jackson.map.deser.std.BaseNodeDeserializer.deserializeArray(JsonNodeDeserializer.java:224)
01-14 13:13:20.103: E/AndroidRuntime(25468): at org.codehaus.jackson.map.deser.std.BaseNodeDeserializer.deserializeObject(JsonNodeDeserializer.java:200)
01-14 13:13:20.103: E/AndroidRuntime(25468): at org.codehaus.jackson.map.deser.std.BaseNodeDeserializer.deserializeArray(JsonNodeDeserializer.java:224)
01-14 13:13:20.103: E/AndroidRuntime(25468): at org.codehaus.jackson.map.deser.std.BaseNodeDeserializer.deserializeObject(JsonNodeDeserializer.java:200)
01-14 13:13:20.103: E/AndroidRuntime(25468): at org.codehaus.jackson.map.deser.std.BaseNodeDeserializer.deserializeArray(JsonNodeDeserializer.java:224)
01-14 13:13:20.103: E/AndroidRuntime(25468): at org.codehaus.jackson.map.deser.std.BaseNodeDeserializer.deserializeObject(JsonNodeDeserializer.java:200)
01-14 13:13:20.103: E/AndroidRuntime(25468): at org.codehaus.jackson.map.deser.std.BaseNodeDeserializer.deserializeObject(JsonNodeDeserializer.java:197)
01-14 13:13:20.103: E/AndroidRuntime(25468): at org.codehaus.jackson.map.deser.std.BaseNodeDeserializer.deserializeArray(JsonNodeDeserializer.java:224)
01-14 13:13:20.103: E/AndroidRuntime(25468): at org.codehaus.jackson.map.deser.std.BaseNodeDeserializer.deserializeObject(JsonNodeDeserializer.java:200)
01-14 13:13:20.103: E/AndroidRuntime(25468): at org.codehaus.jackson.map.deser.std.BaseNodeDeserializer.deserializeObject(JsonNodeDeserializer.java:197)
01-14 13:13:20.103: E/AndroidRuntime(25468): at org.codehaus.jackson.map.deser.std.JsonNodeDeserializer.deserialize(JsonNodeDeserializer.java:58)
01-14 13:13:20.103: E/AndroidRuntime(25468): at org.codehaus.jackson.map.deser.std.JsonNodeDeserializer.deserialize(JsonNodeDeserializer.java:15)
01-14 13:13:20.103: E/AndroidRuntime(25468): at org.codehaus.jackson.map.ObjectMapper._readMapAndClose(ObjectMapper.java:2732)
01-14 13:13:20.103: E/AndroidRuntime(25468): at org.codehaus.jackson.map.ObjectMapper.readValue(ObjectMapper.java:1909)
01-14 13:13:20.103: E/AndroidRuntime(25468): at com.sarla.smartglance.communication.JsonDecoder.decodeResponse(JsonDecoder.java:87)
01-14 13:13:20.103: E/AndroidRuntime(25468): at com.sarla.smartglance.communication.JsonDecoder.decode(JsonDecoder.java:68)
01-14 13:13:20.103: E/AndroidRuntime(25468): at com.sarla.smartglance.communication.MHttpManager$1.run(MHttpManager.java:86)
我尝试了一切,但找不到任何解决方法来解析android上的大量数据。
答案 0 :(得分:7)
使用7-8 megs的JSON,您使用的树模型通常会使用20 - 50 megs的内存(dom模型的大小是3-5倍,适用于XML和JSON)。无论使用哪个库,你都无法做到这一点:它们都使用List
和Map
来构建树,这是一种重量级的方法。
相反,您应该考虑使用普通旧Java对象(POJO),它将使用更少的内存。为此,您需要建模与您的JSON结构匹配的POJO;不知道结构我不能给出一个例子(如果你在问题上添加样本,我可以),但是要解析的代码类似于另一个答案所引用的GSON代码:
MyValue value = mapper.readValue(json, MyValue.class);
这将适用于Jackson以及许多其他Java JSON库(至少Gson,Genson),也将是更快的使用方法。 JSON树本质上是昂贵且重量级的,不能用于多兆字节内容。
最后,如果您的输入包含一系列项目,则可以采用更好的方法对其进行分割(无论单个项目是JsonNode
还是POJO都可以执行此操作!)。但我不知道你的内容是否就是这样。
答案 1 :(得分:1)
我们在这里使用gson lib,使用上面的代码我们可以获得大于50Mb的文件而没有问题:
public static <T extends Object> T readFile(String caminho_arquivo, Type type) {
GsonBuilder gson_builder = new GsonBuilder();
final SimpleDateFormat sdf_date = new SimpleDateFormat("yyyy-MM-dd");
final SimpleDateFormat sdf_datetime = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss.SSS");
gson_builder.registerTypeAdapter(Date.class, new JsonDeserializer<Date>(){
@Override
public Date deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException {
try {
if (json.getAsJsonPrimitive().getAsString().length() == 10)
return sdf_date.parse(json.getAsJsonPrimitive().getAsString());
else
return sdf_datetime.parse(json.getAsJsonPrimitive().getAsString());
} catch (ParseException e) {
Log.e("JSON", "Erro na deserialização de datas no JSON: " + json.getAsJsonPrimitive().getAsString());
return null;
}
}
});
Gson gson = gson_builder.create();
File fileJSON = new File(caminho_arquivo);
FileReader reader = null;
try {
reader = new FileReader(fileJSON);
return gson.fromJson(reader, type);
} catch (FileNotFoundException e) {
e.printStackTrace();
} finally {
try {
reader.close();
if (fileJSON.exists())
fileJSON.delete();
} catch (IOException e) {
e.printStackTrace();
}
}
return null;
}
试试这个lib,这是一个很好的,我们只在服务器端使用jackson,因为Jackson在Android中比gson慢,至少在我们的测试中。
答案 2 :(得分:1)
尝试并使用:
JsonNode rootParser = mapper.readTree(is);
代替。