OkHttp:避免泄漏的连接警告

2022-09-01 03:58:57

我正在使用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();
}

我的应用程序大量使用网络,但该警告经常出现。我从未观察到由这种假定的泄漏引起的任何问题,但我仍然想知道我是否做错了什么

谁能对此有所了解?


答案 1

通过升级到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();
}

答案 2

如其他答案所述,您必须关闭响应。一种稍微简单的方法是在 try 块中声明 ,以便它将自动关闭。ResponseBody

try(ResponseBody body = ....){
....
}

推荐