Android复盘OkHttp HttpLoggingInterceptor造成的OOM

转载请标明出处:http://blog.csdn.net/zhaoyanjun6/article/details/121903763
本文出自【赵彦军的博客】

前言

最近项目中发生了很多 oom,通过 oom 日志分析来看,都跟 okhttp 有关。经过艰难的分析定位,终于发现了造成 oom 的凶手HttpLoggingInterceptor

原因是 HttpLoggingInterceptor 作为 okhttp 请求的过程的日志输出工具, 会把 ResponseBody 里面内容全部读出来,放在内存里。对于普通的网络请求,影响并不大。但是对于下载来说,它会把整个文件读出来,放在内存,就非常容易造成 oom

HttpLoggingInterceptor 简介

该拦截器用于记录应用中的网络请求的信息。

使用:

val okHttpClientBuilder: OkHttpClient.Builder = OkHttpClient.Builder()
        
//创建日志拦截器
val logging = HttpLoggingInterceptor { message -> UtilLog.v("OkHttp", message) }
logging.level = HttpLoggingInterceptor.Level.BODY

//添加日志拦截器
okHttpClientBuilder.addInterceptor(logging)

日志包含4个等级,含四个级别:NONE、BASIC、HEADER、BODY

NONE 不记录

BASIC 请求/响应行
--> POST /greeting HTTP/1.1 (3-byte body)
<-- HTTP/1.1 200 OK (22ms, 6-byte body)

HEADER 请求/响应行 +--> Host: example.com
Content-Type: plain/text
Content-Length: 3
<-- HTTP/1.1 200 OK (22ms)
Content-Type: plain/text
Content-Length: 6

BODY 请求/响应行 ++

下载文件请求阶段需要等很久以及OOM

正常请求没什么问题,但是当我使用这个 okhttp 实例去下载文件的时候出现了问题,请求很久才能拿到正文开始下载。

开始我以为是链接的问题,换了之后发现还是这样,经过一番定位(一行行注释),把问题定位到了HttpLoggingInterceptor拦截器上,发现添加了这个拦截器,并且把level设置为Body的时候就会出现这种情况(由于android不允许在主线程请求网络,所以UI没卡),之后经过调试,发现当代码走到以下位置时,就会卡住。

在这里插入图片描述
大致原因应该是这里需要将内容全部请求出来。

然而在你下载东西的时候,内容是很多的,那么这里的卡住的时长和文件大小成正比,如果文件太大,内存不够,那么系统会不停地GC去获取更多内存,但是总内存就那么点大,所以这就是个死结,甚至会OOM

当然,这里设置level < Body就好了,但是有些时候会有特殊需求,比如说新加一个拦截器,需要根据服务端返回的数据做不同的处理,这里就要参照 HttpLoggingInterceptor 自定义一个拦截器,buffer复制的地方必定会经过,那么上面的问题就会出现。

解决办法:

  • 方法一:就是定义两个 okhttpretrofit也是两个)实例,一个专门用来下载文件,不加这个拦截器。另外一个专门添加这个拦截器去处理服务器返回数据。

  • 方法二:设置 level < Body

  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
Android Studio中使用OkHttp3需要进行以下步骤: 首先,在项目的app/build.gradle文件中添加OkHttp库的依赖,可以使用以下代码添加: implementation 'com.squareup.okhttp3:okhttp:3.14.2' 同时,还需要添加OkHttp需要依赖的基础库Okio,可以使用以下代码添加: implementation 'com.squareup.okio:okio:1.17.4' 接下来,在activity_main.xml文件中添加一个Button按钮和一个TextView控件,可以使用以下代码添加: <Button android:id="@id/send_request" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="Send request get" /> <TextView android:id="@id/request_text" android:layout_width="match_parent" android:layout_height="match_parent" /> 最后,为了保证网络连接和文件读写的权限,需要在AndroidManifest.xml文件中添加以下权限: <!-- 连接网络 --> <uses-permission android:name="android.permission.INTERNET" /> <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" /> <!-- 读写文件 --> <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/> 以上是在Android Studio中使用OkHttp3的一般步骤和权限设置。使用OkHttp3可以实现网络请求和数据传输的功能。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* [Android Studio的OkHttp3使用](https://blog.csdn.net/m0_52664896/article/details/123979466)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *3* [Android Studio 网络编程 OkHttp3 用法](https://blog.csdn.net/moniteryao/article/details/52683691)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值