从任务中抛出聚合异常不会破坏所有其他任务

时间:2013-10-14 13:02:07

标签: c# task

我有两个任务:

try
{
    List<Task> tasks = new List<Task>();

    Task task1 = Task.Factory.StartNew(() => func(param));
    tasks.Add(task1);
    if (someCondition)
    {
        Task task2 = Task.Factory.StartNew(() => func2(param2));
        tasks.Add(task2);
     }

     tasksArr = tasks.ToArray();

     Task.WaitAll(tasksArr);
 }
 catch (AggregateException e)
 {
     //handle
 }
 catch (Exception)
 {
     //handle
 }

如果其中一个任务发生异常,我会抛出一个新的聚合异常,但只有在第二个任务完成后才会到达catch语句。我想立即抓住它,以便在第二项任务中停止不必要的工作。

我尝试使用取消令牌:

CancellationTokenSource cts = new CancellationTokenSource();

Task.WaitAll(tasksArr,cts.Token);

但这不是一个好的解决方案,因为它要求处理内部异常+被捕获的异常将是OperationCanceledException

有什么想法吗?

1 个答案:

答案 0 :(得分:0)

要取消等待,必须将CancellationTokenSource类的实例传递给Task.WaitAll Method (Task[], CancellationToken)。此外,对任务失败时运行的每个任务使用 continuation (错误,即TaskStatus.Faulted)。

var cancellationTokenSource = new CancellationTokenSource();

var tasks = new List<Task>();

var firstTask = new Task(FirstMethod);
firstTask.ContinueWith(t =>
    {
        Console.WriteLine("First task exception: {0}", t.Exception);
        cancellationTokenSource.Cancel();
    }, CancellationToken.None, TaskContinuationOptions.OnlyOnFaulted, TaskScheduler.Default);
tasks.Add(firstTask);
firstTask.Start();

var secondTask = new Task(SecondMethod);
secondTask.ContinueWith(t =>
    {
        Console.WriteLine("Second task exception: {0}", t.Exception);
        cancellationTokenSource.Cancel();
    }, CancellationToken.None, TaskContinuationOptions.OnlyOnFaulted, TaskScheduler.Default);
tasks.Add(secondTask);
secondTask.Start();

try
{
    var cancellationToken = cancellationTokenSource.Token;
    var tasksArray = tasks.ToArray();
    Task.WaitAll(tasksArray, cancellationToken);
}
catch (Exception e)
{
    // ...
}