【问题标题】:How to fix OutOfMemoryError thrown while trying to throw OutOfMemoryError with retrofit如何修复在尝试通过改造抛出 OutOfMemoryError 时抛出的 OutOfMemoryError
【发布时间】:2017-10-28 09:46:51
【问题描述】:

我正在使用改造在我的应用程序中下载视频、mp3、jpg、pdf 等媒体文件。当我想下载 55MB 的 mp4 格式的大文件时出现问题。当我想下载此文件时,我收到如下错误:

OutOfMemoryError threw while trying to throw OutOfMemoryError; no stack trace available

这是我的代码:

  private void downloadFile() {

    ArrayList<FileModel> filesInDB = G.bootFileFromFileDB();

    for (final FileModel fm : filesInDB) {

      APIService downloadService = ServiceGenerator.createServiceFile(APIService.class, "username", "password");

      //Id of apk file that you want to download
      Call<ResponseBody> call = downloadService.downloadFileWithDynamicUrlSync("file/download/" + String.valueOf(fm.getFileId()));
      call.enqueue(new Callback<ResponseBody>() {

        @Override
        public void onResponse(Call<ResponseBody> call, Response<ResponseBody> response) {
          if (response.isSuccess()) {

            Log.d("LOGOO", "server contacted and has file");

            boolean writtenToDisk = writeResponseBodyToDisk(response.body(), fm.getFileName(), fm.getFileExtension());

            response = null;


            Log.d("LOGOO", "file download was a success? " + writtenToDisk);


          } else {
            Log.d("LOGOO", "server contact failed");
          }
        }

        @Override
        public void onFailure(Call<ResponseBody> call, Throwable t) {
          Log.i("LOGO", "Error is : " + t.getMessage());

          Toast.makeText(ActivityInternet.this, R.string.internet_error, Toast.LENGTH_LONG).show();
          Intent intent = new Intent(ActivityInternet.this, ActivityStartup.class);
          startActivity(intent);
        }
      });
    }

这是我正在使用的“writeResponseBodyToDisk”方法:

  private boolean writeResponseBodyToDisk(ResponseBody body, String fileName, String fileExtension) {

    try {

      // Location to save downloaded file and filename
      File futureStudioIconFile = new File(G.DIR_APP + fileName + fileExtension);
      InputStream inputStream = null;
      OutputStream outputStream = null;
      try {
        byte[] fileReader = new byte[4096];
        long fileSize = body.contentLength();
        long fileSizeDownloaded = 0;
        inputStream = body.byteStream();
        outputStream = new FileOutputStream(futureStudioIconFile);
        while (true) {
          int read = inputStream.read(fileReader);
          if (read == -1) {
            break;
          }
          outputStream.write(fileReader, 0, read);
          fileSizeDownloaded += read;
          Log.d("LOGO", "file download: " + fileSizeDownloaded + " of " + fileSize);
        }
        outputStream.flush();
        return true;
      } catch (IOException e) {
        return false;
      } finally {
        if (inputStream != null) {
          inputStream.close();
        }
        if (outputStream != null) {
          outputStream.close();
        }
      }
    } catch (IOException e) {
      return false;
    }


  }

最后这是我的 createServiceFile 方法:

public static <S> S createServiceFile(Class<S> serviceClass, String username, String password) {
        if (username != null && password != null) {

            String credentials = username + ":" + password;
            final String basic =
              "Basic " + Base64.encodeToString(credentials.getBytes(), Base64.NO_WRAP);
            httpClient.addInterceptor(new Interceptor() {
                @Override
                public Response intercept(Chain chain) throws IOException {
                    Request original = chain.request();
                    Request.Builder requestBuilder = original.newBuilder()
                      .header("Authorization", basic)
                      .header("Accept", "text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,audio/mp4,image/jpeg,*/*;q=0.8")
                      .method(original.method(), original.body());
                    Request request = requestBuilder.build();
                    return chain.proceed(request);
                }
            });
        }
        OkHttpClient client = httpClient.build();
        Retrofit retrofit = builder.client(client).build();
        return retrofit.create(serviceClass);
    }

如果你能帮助我,我真的很感激:)

【问题讨论】:

  • 这可能会有所帮助:futurestud.io/tutorials/…
  • 你在清单中添加了这个 android:largeHeap="true" 吗?
  • @quicklearner 只有在确实需要内存时才应该这样做。这里不是这种情况,因为可以使用流式传输。 IE。在完全加载之前将文件的第一个字节写入磁盘。

标签: android out-of-memory retrofit


【解决方案1】:

在改造中处理大文件下载时需要注意 4 件事:

  1. 确保在AndroidManifest.xml 中使用android:largeHeap="true" 作为&lt;application&gt; 的属性。
  2. 确保您使用的是来自 Retrofit 的 @Streaming 注释,以便流式传输解决方案而不是整体读取它,这会占用内存。
  3. 使用AsyncTask 处理响应,如this link 中所述。在您的情况下,这意味着从 AsyncTask 中调用 writeResponseBodyToDisk
  4. 最后要避免的是使用Level.BODY okhttp3 日志拦截器。将它与流式响应一起使用仍将整个响应主体保留在内存中,从而抵消了改造提供的 @Streaming 支持的优势。

【讨论】:

  • 我不明白数字 4,但我在我的应用程序中做了 1、2、3,现在我可以下载所有文件,但有时,有些文件下载的大小不同(比原始文件小) 从服务器
  • 数字 4 仅在您使用日志拦截器来记录您使用改造发送的请求/响应的情况下。你目前没有这样做。为了理解为什么你会得到部分数据,你可以尝试: 1. 添加一个日志拦截器并找出是客户端(改造)还是你的服务器负责数据丢失。在this answer 中查找说明。 2. 检查您的网络服务器中的access.log(如果可以的话),看看它是否发送了完整的数据。最后去掉拦截器很重要!可能会导致OOM!
  • @sheba ,我们正在合作并使用 TomEE 作为 Web 服务器。这样的情况有什么问题吗?
  • 谢谢@sheba!由于高处理和刚刚解决的提示 1,我在特定活动中旋转屏幕时遇到了问题!
猜你喜欢
  • 2018-09-02
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2012-02-13
  • 1970-01-01
  • 2014-12-31
相关资源
最近更新 更多