jga*_*fin 3 c# task-parallel-library
我有以下方法:
public async Task PublishAsync<TApplicationEvent>(TApplicationEvent e)
where TApplicationEvent : ApplicationEvent
{
using (var scope = _container.CreateScope())
{
var implementations = scope.ResolveAll<IApplicationEventSubscriber<TApplicationEvent>>();
var tasks = implementations.Select(x => x.HandleAsync(e));
try
{
await Task.WhenAll(tasks);
EventPublished(this, new EventPublishedEventArgs(scope, e, true));
}
catch
{
EventPublished(this, new EventPublishedEventArgs(scope, e, false));
throw;
}
}
}
Run Code Online (Sandbox Code Playgroud)
我认为所有任务都将在抛出异常之前执行,但似乎该方法在第一个任务抛出异常时中止.
我可以配置WhenAll
执行所有任务并在返回之前生成包含所有失败的AggregateException吗?
这正是WhenAll
做了什么.但是,当您await
抛出一个在其中有多个异常的聚合异常的任务时,它将仅重新抛出聚合异常中的第一个异常,而不是重新抛出聚合异常本身.(这是因为抛出异常的绝大多数任务在聚合中永远不会有多个表达式,因此解除异常几乎总是所需的行为.)
只需保留对它Task
之前的引用,await
以便以后可以访问聚合异常,或者只是不await
使用结果WhenAll
而是使用手动延续.
Task.WhenAll() -> 创建一个任务,该任务将在所有提供的任务完成后完成。
无论单个任务中发生什么,它将完成所有任务的执行。
给定任务集合中的任务之一出现异常不会停止其他任务的执行。
var tasks = new List<Task>();
Task t1 = Task.Run(() =>
{
throw new Exception("test1");
});
tasks.Add(t1);
Task t2 = Task.Run(() =>
{
throw new Exception("test2");
});
tasks.Add(t2);
Task t3 = Task.Run(() =>
{
Console.WriteLine("Pseudo Work");
});
tasks.Add(t3);
try
{
await Task.WhenAll(tasks);
}
catch (Exception ex)
{
// Only t1 & t2 will qualify and t3 will be successful (Pseudo Work will be printed on console)
var exceptions = tasks.Where(t => t.Exception != null).Select(t => t.Exception);
}
Run Code Online (Sandbox Code Playgroud)