使用HttpClient异步方法时内存泄漏c#

时间:2017-12-31 14:06:55

标签: c# memory-leaks async-await httpclient

以下代码每秒调用多次。如果_client.PostJsonAsync抛出OperationCanceledException(例如由于超时),则返回Task.FromResult(default(T)),但是如果发生这种情况,则内存使用(和CPU使用率)会快速开始爬升,直到抛出OutOfMemoryExceptions(甚至当没有抛出异常时,内存似乎总是爬得越来越慢。代码(或其过度使用)是否有任何可能导致内存泄漏的问题?

{{1}}

1 个答案:

答案 0 :(得分:3)

我可以指出两个使你的记忆力下降的事情:

  • 正如Nkosi在评论中指出的PostAsJsonAsync返回类型是HttpResponseMessage,它实现了IDisposable所以你应该处理它,只需在它周围添加一个using语句。
    这就是您的代码泄漏的地方,以及即使没有抛出异常也会导致内存使用率上升的原因。

  • 当抛出异常时,你的内存使用率似乎上升得更快的原因是,除了你在这个场景中的第一个问题,你使用Task.FromResult创建了很多新任务,这会增加你的压力内存使用情况 虽然它似乎没有泄漏它仍然会使GC更加努力,这就是为什么我建议你缓存你返回的任务(你可以重用相同的任务实例,因为你总是返回相同的值),甚至更好的不创建新任务。
    由于您的方法已经标记为async,您可以立即返回default(T),编译器将为您生成状态机并将其包装在任务中(无论如何它都会执行,没有理由创建并等待一项新任务。)

您修改后的代码应如下所示:

private async Task<T> GetResponse<T>(Dictionary<string, object> args, string method)
{
    try
    {
        using(var response = await _client.PostAsJsonAsync(method, args))
        {
            return await Deserialise<T>(response.Content); 
        }
    }
    catch (OperationCanceledException)
    {
        return default(T);
    }
}