我需要在控制台应用程序中运行多个异步任务,并等待它们全部完成后再进行进一步处理。
有很多文章,但我似乎越读越糊涂。我已经阅读并理解了 Task 库的基本原理,但显然我在某个地方丢失了一个链接。
我知道可以链式执行任务,使它们在另一个任务完成后开始(这基本上是我读过的所有文章的情况),但我希望我的所有任务同时运行,并且我想知道何时它们都已完成。
像这种情况下最简单的实现方式是什么?
我需要在控制台应用程序中运行多个异步任务,并等待它们全部完成后再进行进一步处理。
有很多文章,但我似乎越读越糊涂。我已经阅读并理解了 Task 库的基本原理,但显然我在某个地方丢失了一个链接。
我知道可以链式执行任务,使它们在另一个任务完成后开始(这基本上是我读过的所有文章的情况),但我希望我的所有任务同时运行,并且我想知道何时它们都已完成。
像这种情况下最简单的实现方式是什么?
两个回答都没有提到可等待的Task.WhenAll:
var task1 = DoWorkAsync();
var task2 = DoMoreWorkAsync();
await Task.WhenAll(task1, task2);
Task.WaitAll
和 Task.WhenAll
的主要区别在于前者会阻塞(类似于对单个任务使用 Wait
),而后者不会阻塞且可被等待,直到所有任务完成才将控制权交回给调用方。此外,异常处理也有所不同:
Task.WaitAll
:
如果至少有一个任务被取消或在执行过程中抛出了异常,则返回的 AggregateException 将包含 InnerExceptions 集合中的 OperationCanceledException。
Task.WhenAll
:
如果任何一个提供的任务处于 faulted 状态,则返回的任务也会以 Faulted 状态结束,其中的异常将包含每个提供的任务中解包后的异常集合的聚合。
如果没有提供的任务 faulted,但至少有一个被取消,则返回的任务将以 Canceled 状态结束。
如果没有任务 faulted 且没有任务被取消,则结果任务将以 RanToCompletion 状态结束。
如果提供的数组/枚举不包含任务,则返回的任务将立即转换为 RanToCompletion 状态,然后返回给调用方。你可以创建许多任务,例如:
List<Task> TaskList = new List<Task>();
foreach(...)
{
var LastTask = new Task(SomeFunction);
LastTask.Start();
TaskList.Add(LastTask);
}
Task.WaitAll(TaskList.ToArray());
WhenAll
返回一个可等待的Task
或WaitAll
,没有返回类型并将阻止进一步的代码执行,类似于Thread.Sleep
,直到所有任务完成、取消或故障。WhenAll | WaitAll | |
---|---|---|
任何已提供任务以错误状态完成 | 将返回一个带有错误状态的任务。异常将包含从每个已提供任务中解压缩异常集合的聚合结果。 | 将抛出一个AggregateException 。 |
未提供的任务中没有任何任务失败,但至少有一个被取消 | 返回的任务将以TaskStatus.Canceled 状态结束 | 将抛出一个包含在其InnerExceptions 集合中有一个OperationCanceledException 的AggregateException 。 |
给出了一个空列表 | 将抛出一个ArgumentException | 返回的任务将立即转换为TaskStatus.RanToCompletion 状态,然后返回给调用方。 |
不会阻塞当前线程 | 会阻塞当前线程 |
示例
var tasks = new Task[] {
TaskOperationOne(),
TaskOperationTwo()
};
Task.WaitAll(tasks);
// or
await Task.WhenAll(tasks);
如果您想以特定的顺序运行任务,您可以从这个答案中获取灵感。
await
,同时又使用WaitAll
或WhenAll
呢?Task[]
初始化中的任务不应该没有await
吗? - dee zgpublic static Task ForEachAsync<T>(this IEnumerable<T> sequence, Func<T, Task> action) {
return Task.WhenAll(sequence.Select(action));
}
这样调用:
await sequence.ForEachAsync(item => item.SomethingAsync(blah));
或者使用异步lambda函数:
await sequence.ForEachAsync(async item => {
var more = await GetMoreAsync(item);
await more.FrobbleAsync();
});
public static Task ForEachAsync(this IEnumerable sequence, Func> action)
{
return Task.WhenAll(sequence.Select(action));
}
- Brian Davis又是一个答案……但通常情况下,我发现自己需要同时加载数据并将其放入变量中,例如:
var cats = new List<Cat>();
var dog = new Dog();
var loadDataTasks = new Task[]
{
Task.Run(async () => cats = await LoadCatsAsync()),
Task.Run(async () => dog = await LoadDogAsync())
};
try
{
await Task.WhenAll(loadDataTasks);
}
catch (Exception ex)
{
// handle exception
}
LoadCatsAsync()
和 LoadDogAsync()
只是数据库调用,那么它们是 IO-bound。Task.Run()
适用于 CPU-bound 工作;如果你只是在等待来自数据库服务器的响应,它会增加额外的不必要开销。对于 IO-bound 工作,Yuval 的被接受的答案是正确的方式。 - Stephen Kennedy您是否希望将Task
链接起来,或者它们可以以并行方式调用?
对于链接
只需执行以下操作:
Task.Run(...).ContinueWith(...).ContinueWith(...).ContinueWith(...);
Task.Factory.StartNew(...).ContinueWith(...).ContinueWith(...).ContinueWith(...);
不要忘记在每个ContinueWith
中检查之前的Task
实例,因为它可能会出现故障。
对于并行方式
我遇到的最简单的方法是:Parallel.Invoke
否则可以使用Task.WaitAll
,或者甚至可以使用WaitHandle
进行倒计时以达到零操作(等等,还有一个新类:CountdownEvent
),或者...
var tasks = new Func<Task>[]
{
() => myAsyncWork1(),
() => myAsyncWork2(),
() => myAsyncWork3()
};
await Task.WhenAll(tasks.Select(task => task()).ToArray()); //Async
Task.WaitAll(tasks.Select(task => task()).ToArray()); //Or use WaitAll for Sync
Use this... Instead of this... When wishing to do this...
await Task.Wait
Task.Result Retrieving the result of a background task
await Task.WhenAny Task.WaitAny Waiting for any task to complete
await Task.WhenAll Task.WaitAll Waiting for all tasks to complete
await Task.Delay Thread.Sleep Waiting for a period of time
public async Task DoSeveralThings()
{
// Start all the tasks
Task first = DoFirstThingAsync();
Task second = DoSecondThingAsync();
// Then wait for them to complete
var firstResult = await first;
var secondResult = await second;
}
第二个
任务完成之前,第一个
任务失败导致任务丢失。正确的等待多个任务的方法是使用Task.WhenAll
方法:await Task.WhenAll(first, second);
,然后你可以逐个await
它们以获取结果,因为你知道所有的任务都已成功完成了。 - Theodor Zouliassecond
的失败也应该泄漏first
。但是您的代码并不支持这一点。它的泄漏行为是不对称的。 - Theodor Zoulias // method to run tasks in a parallel
public async Task RunMultipleTaskParallel(Task[] tasks) {
await Task.WhenAll(tasks);
}
// methode to run task one by one
public async Task RunMultipleTaskOneByOne(Task[] tasks)
{
for (int i = 0; i < tasks.Length - 1; i++)
await tasks[i];
}
// method to run i task in parallel
public async Task RunMultipleTaskParallel(Task[] tasks, int i)
{
var countTask = tasks.Length;
var remainTasks = 0;
do
{
int toTake = (countTask < i) ? countTask : i;
var limitedTasks = tasks.Skip(remainTasks)
.Take(toTake);
remainTasks += toTake;
await RunMultipleTaskParallel(limitedTasks.ToArray());
} while (remainTasks < countTask);
}
await Task.WhenAll(task1, task2);
前需要单独启动每个任务吗? - ZapnologicaTask.WhenAll
并дёҚдјҡдёәдҪ еҗҜеҠЁд»»еҠЎпјҢдҪ йңҖиҰҒжҸҗдҫӣвҖңзғӯвҖқд»»еҠЎпјҢд№ҹе°ұжҳҜе·Із»ҸејҖе§Ӣжү§иЎҢзҡ„д»»еҠЎгҖӮ - Yuval ItzchakovStartNew
和启动新任务与异步等待它们有什么关系? - Yuval Itzchakov