【问题标题】:Out of memory exception in gson.fromJson()gson.fromJson() 中的内存不足异常
【发布时间】:2012-05-26 11:30:59
【问题描述】:

我使用以下代码将 Json 字符串(strWebserviceResult)转换为我的对象:

EntMyClass entMyClass = gson.fromJson(strWebserviceResult,EntMyClass.class);

strWebserviceResult 很大(大约 2.5 MB)时,我在 Android 手机设备上的这条线上得到 Out of memory 异常,而不是在具有更大内存的平板电脑中。 p>

我该如何解决。

有人有什么建议吗?

05-26 15:52:49.607: E/dalvikvm-heap(2078): Out of memory on a 9200-byte allocation.
05-26 15:52:49.618: E/dalvikvm(2078): Out of memory: Heap Size=31879KB, Allocated=27693KB, Bitmap Size=936KB, Limit=32768KB
05-26 15:52:49.618: E/dalvikvm(2078): Extra info: Footprint=31879KB, Allowed Footprint=31879KB, Trimmed=7400KB
05-26 15:52:49.618: E/AndroidRuntime(2078): FATAL EXCEPTION: Thread-19
05-26 15:52:49.618: E/AndroidRuntime(2078): java.lang.OutOfMemoryError: (Heap Size=31879KB, Allocated=27693KB, Bitmap Size=936KB)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at java.util.ArrayList.add(ArrayList.java:123)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.DefaultTypeAdapters$CollectionTypeAdapter.deserialize(DefaultTypeAdapters.java:664)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.DefaultTypeAdapters$CollectionTypeAdapter.deserialize(DefaultTypeAdapters.java:624)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.JsonDeserializerExceptionWrapper.deserialize(JsonDeserializerExceptionWrapper.java:51)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.JsonDeserializationVisitor.invokeCustomDeserializer(JsonDeserializationVisitor.java:92)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.JsonObjectDeserializationVisitor.visitFieldUsingCustomHandler(JsonObjectDeserializationVisitor.java:117)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.ReflectingFieldNavigator.visitFieldsReflectively(ReflectingFieldNavigator.java:63)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.ObjectNavigator.accept(ObjectNavigator.java:120)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.JsonDeserializationContextDefault.fromJsonObject(JsonDeserializationContextDefault.java:76)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.JsonDeserializationContextDefault.deserialize(JsonDeserializationContextDefault.java:54)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.DefaultTypeAdapters$CollectionTypeAdapter.deserialize(DefaultTypeAdapters.java:663)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.DefaultTypeAdapters$CollectionTypeAdapter.deserialize(DefaultTypeAdapters.java:624)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.JsonDeserializerExceptionWrapper.deserialize(JsonDeserializerExceptionWrapper.java:51)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.JsonDeserializationVisitor.invokeCustomDeserializer(JsonDeserializationVisitor.java:92)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.JsonDeserializationVisitor.visitUsingCustomHandler(JsonDeserializationVisitor.java:80)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.ObjectNavigator.accept(ObjectNavigator.java:101)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.JsonDeserializationContextDefault.fromJsonArray(JsonDeserializationContextDefault.java:67)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.JsonDeserializationContextDefault.deserialize(JsonDeserializationContextDefault.java:52)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.Gson.fromJson(Gson.java:551)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.Gson.fromJson(Gson.java:498)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.Gson.fromJson(Gson.java:467)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.Gson.fromJson(Gson.java:417)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at com.google.gson.Gson.fromJson(Gson.java:389)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at org.mabna.order.businessLayer.BoWebService.getDataForUpdate(BoWebService.java:188)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at org.mabna.order.ui.ActToolDataExchange.threadGetDataForFullUpdate(ActToolDataExchange.java:371)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at org.mabna.order.ui.ActToolDataExchange.access$9(ActToolDataExchange.java:362)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at org.mabna.order.ui.ActToolDataExchange$33.run(ActToolDataExchange.java:603)
05-26 15:52:49.618: E/AndroidRuntime(2078):     at org.mabna.order.utils.Utilities$5.run(Utilities.java:778)

【问题讨论】:

  • 也许this可以帮助你

标签: android json gson out-of-memory


【解决方案1】:

尝试使用将 JsonReader 作为输入的fromJson 方法。这应该允许您一次不需要整个输入字符串都在内存中。下面是一些示例代码:

        final HttpURLConnection c = (HttpURLConnection) url.openConnection();
        final InputStreamReader isr = new InputStreamReader(c.getInputStream());
        final JsonReader reader = new JsonReader(isr);
        final EntMyClass entMyClass = GSON.fromJson(reader, EntMyClass.class);
        reader.close();
        c.disconnect();

【讨论】:

  • 我的初始数据是一个加密字符串。我将其解密为字符串,并将 strWebserviceResult 作为字符串。如何将 strWebserviceResult 转换为 InputStream
  • 加密字符串 -> 将 strWebserviceResult 清除为 json 字符串 -> 我的对象
  • 也许您可以先将解密后的字符串存储到本地文件中?然后您应该能够从本地文件中获取 InputStream。
【解决方案2】:

与其他帖子一样,我想问是否有任何方法可以避免您的应用使用大量内存。如果你能做到这一点,那将是最优化的解决方案。如果您的应用确实需要这么多内存,您可以尝试在清单中为您的应用设置android:largeHeap="true"。这是参考:

http://developer.android.com/reference/android/R.styleable.html#AndroidManifestApplication_largeHeap

观看此视频了解更多信息:

http://www.youtube.com/watch?v=_CruQY55HOk

【讨论】:

  • 不建议使用此解决方案,因为它还会增加 GC 时间(因为要处理的内存更多)
  • 这不是正确的解决方案。增加内存只会延迟OOM,并不能解决。
【解决方案3】:

首先。您真的需要将 2.5MB 数据加载到内存中吗?
首先创建了一个大字符串,然后对其进行解析(内存中的另一个对象),然后创建了一个巨大的 EntMyClass 实例。这种方法确实效率低下。

我猜你在这个对象中有 List。只有您提供的信息,我建议将数据流式传输到数据库中。然后您可以创建适配器以在 ListView 中显示数据。

有很多方法可以做到这一点。我使用的是 Jackson 库,因为它是最快的,GSON 应该有类似的功能。

假设您在此对象中有一个列表,下面是如何在数据库中存储 json 数组的示例:

    //create parser
    final JsonParser jsonParser = objectMapper.getJsonFactory().createJsonParser(inputStream);

    JsonToken jsonToken = jsonParser.nextToken();
    while ((jsonToken = jsonParser.nextToken()) != null && jsonToken != JsonToken.END_ARRAY) {
        //map json object to ItemClass
        final ItemClass item = jsonParser.readValueAs(ItemClass.class);
        //store in database
        itemDAO.insert(item);
    }

并用数据库事务包围,不仅数据会在出错的情况下回滚,而且效率更高。

【讨论】:

  • 答案就是我想要的
【解决方案4】:

请阅读gson作者的答案-here

还有我的 5 美分:

        JsonReader r = null;
        try {
            Reader reader = new BufferedReader(new InputStreamReader(is));
            r = new JsonReader(reader);
            result = gson.fromJson(r, clazz);
        } finally {
            if (null != r) {
                r.close();
            }
        }

【讨论】:

  • 谢谢德米特里。如何将我的字符串转换为 InputStream (is)?
  • 您对该字符串使用哪种加密/解密算法?为什么我要问它?原因是您需要按小块组织处理这个字符串,而不是整个 2.5mb,换句话说 - 部分处理。
  • public String getClearedResponse(String decryptionPassword) throws Exception { String zippedString = Crypto.decrypt(this.content, decryptionPassword); return Utilities.decompress(zippedString); }
  • 它也有解压程序:(
  • 其实应该是这样的: public InputStream getClearedResponse(String decryptionPassword) throws Exception { return new BufferedInputStream(new GZIPInputStream(Crypto.decrypt(is, decryptionPassword))); }。在您的 Crypto.decrypt 中,您需要实现您的CryptoReader。是你自己的加密算法吗?
猜你喜欢
  • 1970-01-01
  • 2010-10-05
  • 1970-01-01
相关资源
最近更新 更多