我有一个计时器,每2秒启动2个任务...我在一个简单的列表中跟踪这些任务,(以便我在停止应用程序时可以等待它们完成)。
任务有效地进入数据库,运行几次更新并完成。
任务本身不会花费超过一秒钟的时间
...
// global variables to keep track of the running tasks.
List<Task> _tasks = new List<Task>();
CancellationTokenSource _cts = new CancellationTokenSource();
// in the timer function
private void Timer(object sender, ElapsedEventArgs e)
{
_tasks.Add( Foo1Async(_cts.Token) );
_tasks.Add( Foo2Async(_cts.Token) );
// remove the completed ones
_tasks.RemoveAll(t => t.IsCompleted);
}
...
几分钟后,内存从40Mb上升到130Mb(并不断攀升)...
如果我仅替换 下面的代码和,并且没有其他内容
...
_tasks.Add( Foo1Async( CancellationToken.None)) );
_tasks.Add( Foo2Async( CancellationToken.None)) );
...
内存保持稳定在40Mb,并且永远不会增加。
几点
System.Timers.Timer
,我只在处理计时器事件后才重新启动它。如果我拍摄内存快照,CancellationCallbackInfo
的数量似乎是问题所在,但我不知道它们来自何处以及如何释放它们。
看看.NET源代码,在CancellationTokenSource
中使用了回调,但我不确定如何向其中添加回调(或如何释放它们)。
关于使用_cts.Token
和使用CancellationToken.None
时可能导致内存泄漏的任何建议
答案 0 :(得分:3)
.NET DBCommand的某些代码中存在/存在问题,异步函数注册为取消,但只有在出现异常时才进行处理。如您所见,在大多数情况下,它只会增加列表。
此问题已在last year in .NET core v2.x某个时间修复,(不确定是否会在.NET标准中修复,我使用的是4.6.1,但这仍然是一个问题)。
解决该问题的一种方法是围绕所需的异步函数编写自己的包装器。
您还可以调用非异步功能,(但您会丢失异步功能为您提供的“取消”功能)。