我正在使用OkHttp 3,并且我一直在泄漏连接警告:
WARNING: A connection to https://help.helpling.com/ was leaked. Did you forget to close a response body?
Jul 14, 2016 6:57:09 PM okhttp3.ConnectionPool pruneAndGetAllocationCount
每当我收到ResponseBody
时,我都会调用.string()
,这可能会为我关闭流,或者我会在finally
块中明确关闭它,方法如下:
ResponseBody responseBody = response.body();
try (Reader responseReader = responseBody.charStream()) {
...
}
finally {
responseBody.close();
}
我的应用程序大量使用网络,但这种警告经常出现。我从未发现过这种假定泄漏引起的任何问题,但我仍然想了解 if 和 我做错了什么。
有人可以对此有所了解吗?
答案 0 :(得分:9)
通过升级到OkHttp 3.7,Eclipse开始警告我潜在的资源泄漏。我发现我的问题出在我写的这个方法中:
public static Response getResponse(HttpUrl url, OkHttpClient client) throws IOException {
Builder request = new Request.Builder().url(url);
Response response = client.newCall(request.build()).execute();
if (!response.isSuccessful()) {
boolean repeatRequest = handleHttpError(response);
if (repeatRequest)
return getResponse(url, client, etag);
else
throw new IOException(String.format("Cannot get successful response for url %s", url));
}
return response;
}
我认为通过始终调用getResponse(url, client).body().string()
,流会自动关闭。但是,只要响应不成功,就会在执行.string()
之前引发异常,因此流将保持打开状态。
在响应失败的情况下添加显式关闭可以解决问题。
if (!response.isSuccessful()) {
boolean repeatRequest = handleHttpError(response);
response.close();
}
答案 1 :(得分:6)
如其他答案中所述,您必须关闭响应。稍微简洁的方法是在try块中声明ResponseBody
,以便它自动关闭。
try(ResponseBody body = ....){
....
}