回答:
Task.WaitAll
すべてが完了するまで現在のスレッドをブロックします。
Task.WhenAll
すべてが完了するまで待機するアクションを表すタスクを返します。
つまり、非同期メソッドから、以下を使用できます。
await Task.WhenAll(tasks);
...つまり、すべてが完了するとメソッドは続行しますが、その時までスレッドを拘束してハングアップすることはありません。
WhenAll
、それはスレッドをブロックと同じではありません。
JonSkeetの答えは、通常は優れた方法で違いを説明していますが、別の違いがあります。例外処理です。
Task.WaitAll
AggregateException
いずれかのタスクがスローしたときにをスローし、スローされたすべての例外を調べることができます。アンラップと「戻り」のみ最初の例外。await
await Task.WhenAll
AggregateException
以下のプログラムを実行しawait Task.WhenAll(taskArray)
て出力すると以下のようになります。
19/11/2016 12:18:37 AM: Task 1 started
19/11/2016 12:18:37 AM: Task 3 started
19/11/2016 12:18:37 AM: Task 2 started
Caught Exception in Main at 19/11/2016 12:18:40 AM: Task 1 throwing at 19/11/2016 12:18:38 AM
Done.
以下のプログラムをTask.WaitAll(taskArray)
出力付きで実行すると以下のようになります。
19/11/2016 12:19:29 AM: Task 1 started
19/11/2016 12:19:29 AM: Task 2 started
19/11/2016 12:19:29 AM: Task 3 started
Caught AggregateException in Main at 19/11/2016 12:19:32 AM: Task 1 throwing at 19/11/2016 12:19:30 AM
Caught AggregateException in Main at 19/11/2016 12:19:32 AM: Task 2 throwing at 19/11/2016 12:19:31 AM
Caught AggregateException in Main at 19/11/2016 12:19:32 AM: Task 3 throwing at 19/11/2016 12:19:32 AM
Done.
プログラム:
class MyAmazingProgram
{
public class CustomException : Exception
{
public CustomException(String message) : base(message)
{ }
}
static void WaitAndThrow(int id, int waitInMs)
{
Console.WriteLine($"{DateTime.UtcNow}: Task {id} started");
Thread.Sleep(waitInMs);
throw new CustomException($"Task {id} throwing at {DateTime.UtcNow}");
}
static void Main(string[] args)
{
Task.Run(async () =>
{
await MyAmazingMethodAsync();
}).Wait();
}
static async Task MyAmazingMethodAsync()
{
try
{
Task[] taskArray = { Task.Factory.StartNew(() => WaitAndThrow(1, 1000)),
Task.Factory.StartNew(() => WaitAndThrow(2, 2000)),
Task.Factory.StartNew(() => WaitAndThrow(3, 3000)) };
Task.WaitAll(taskArray);
//await Task.WhenAll(taskArray);
Console.WriteLine("This isn't going to happen");
}
catch (AggregateException ex)
{
foreach (var inner in ex.InnerExceptions)
{
Console.WriteLine($"Caught AggregateException in Main at {DateTime.UtcNow}: " + inner.Message);
}
}
catch (Exception ex)
{
Console.WriteLine($"Caught Exception in Main at {DateTime.UtcNow}: " + ex.Message);
}
Console.WriteLine("Done.");
Console.ReadLine();
}
}
await t1; await t2; await t3;
VSawait Task.WhenAll(t1,t2,t3);
await
、2つの方法の違いではないと考えています。どちらもAggregateException
、直接またはプロパティ(Task.Exception
プロパティ)を介してスローするを伝播します。
彼らは何をしますか:
違いは何ですか:
次の場合に使用します。
WaitAll
私が理解しているように、あなたにはその可能性はありません。
Task.WaitAll
後の呼び出しを変えてみませんか?つまり、タスクを開始した直後に呼び出すのではありません。