[英]Out of memory error while parsing a large JSON using Jackson library on Android
我正在使用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上的大量數據。
使用7-8兆的JSON,您使用的樹模型通常會使用20 - 50兆的內存(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,都可以完成此操作!)。 但我不知道你的內容是否那樣。
我們在這里使用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更慢,至少在我們的測試中。
嘗試並使用:
JsonNode rootParser = mapper.readTree(is);
代替。
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.