基于return resutl在TPL中取消多个任务

时间:2016-05-17 09:33:30

标签: c# task-parallel-library task cancellation-token

我有一个场景,我有多个函数可以并行调用,我正在使用TPL执行该任务。我用过

ConcurrentDictionary<string, List<result>> dictionary = new ConcurrentDictionary<string, List<result>>();

var tasks = Task.Run(()=>
{
  new Task(()=> {dictionary.TryAdd("First", CallFirstFunction());});
  new Task(()=> {dictionary.TryAdd("Second", CallSecondFunction());});
  new Task(()=> {dictionary.TryAdd("Third", CallThirdFunction());});
  new Task(()=> {dictionary.TryAdd("Fourth", CallFourthFunction());});
});

现在,我需要等到所有函数都被执行并在并发字典中返回一些结果,这样我就可以用它来进行进一步的处理了。但是如果一些返回的结果为空而不管任务顺序,我也想取消任何任务。如果任何函数返回空结果,我需要同时取消所有剩余的任务。我已经检查了“CancellationToken”类,但我很困惑在任务中使用它以及如何在任务中共享功能状态。

任何帮助将不胜感激。 最诚挚的问候

1 个答案:

答案 0 :(得分:0)

要支持取消,您需要使用CancellationTokenSource。见https://msdn.microsoft.com/en-us/library/mt674886.aspx

您需要做的是将CancellationTokenSource传递给您的方法,这样当结果为空时,您可以致电CancellationTokenSource.Cancel()

async Task DoWorkAsync()
{
    var cts = new CancellationTokenSource();
    var ct = cts.Token;

    var tasks = new List<Task>
    {
        Task.Run(() => AddToDictionary("First", () => CallFirstFunction(), cts), ct),
        Task.Run(() => AddToDictionary("Second", () => CallSecondFunction(), cts), ct),
        Task.Run(() => AddToDictionary("Third", () => CallThirdFunction(), cts), ct),
        Task.Run(() => AddToDictionary("Fourth", () => CallFourthFunction(), cts), ct),
    };

    try
    {           
         await Task.WhenAll(tasks); 
    }
    catch (OperationCanceledException ex)
    {
    }
}

private void AddToDictionary(string key, Func<List<int>> method, CancellationTokenSource cts)
{
    if(cts.IsCancellationRequested)
        return;

    var result = method.Invoke();
    if(!result.Any());
        cts.Cancel();

    if(!cts.IsCancellationRequested)
        dictionary.TryAdd(key, result);
}

此示例假定您调用的操作(CallFirstFunction等)不是异步(不返回任务或任务)。否则,您也应将CancellationToken传递给他们,并使用ct.ThrowIfCancellationRequested

检查取消

取消任务会生成您需要处理的OperationCanceledException。

当然,当任何方法在另一个方法返回空结果之前已经完成时,那些结果将已经存在于字典中。如果您需要其他行为,此示例将为您提供有关CancellationTokenSource如何与Tasks和CancellationTokens相关的基本概念