我对TPL有一些问题。
如您所见,我正在创建2个简单任务并将它们添加到列表中。
问题(预期的行为)是任务在点击" await"之后立即返回。在WorkMethodAsync
中,因此呈现Task.WhenAll
无用。
有没有办法等待子任务完成?
我发现的唯一解决方法是伪造WorkMethodAsync
同步。
private async void button1_Click(object sender, EventArgs e)
{
Tasks = new List<Task>();
var myTask1 = Task.Factory.StartNew(async () => await WorkMethodAsync(), CancellationToken.None, TaskCreationOptions.LongRunning, TaskScheduler.Default);
var myTask2 = Task.Factory.StartNew(async () => await WorkMethodAsync(), CancellationToken.None, TaskCreationOptions.LongRunning, TaskScheduler.Default);
Tasks.Add(myTask1);
Tasks.Add(myTask2);
await Task.WhenAll(Tasks.ToArray());
}
private async Task WorkMethodAsync()
{
while (true)
{
await Task.Delay(10000);
}
return;
}
答案 0 :(得分:5)
问题是您使用的是Task.Factory.StartNew
而不是Task.Run
。
StartNew
是在async-await之前构建的,因此不能很好地支持它。在这种情况下,由于您的委托返回一个任务,StartNew
创建一个任务,返回值为Task<Task>>
。您可以使用Unwrap
来获取代表整个异步操作的任务:
Task<Task> myTask1 = Task.Factory.StartNew(async () => await WorkMethodAsync(), CancellationToken.None, TaskCreationOptions.LongRunning, TaskScheduler.Default);
Task<Task> myTask2 = Task.Factory.StartNew(async () => await WorkMethodAsync(), CancellationToken.None, TaskCreationOptions.LongRunning, TaskScheduler.Default);
Tasks.Add(myTask1.Unwrap());
Tasks.Add(myTask2.Unwrap());
但是,由于Task.Factory.StartNew
对异步方法没有意义,所以我没有理由使用TaskCreationOptions.LongRunning
开头(我的博客上有更多内容:{{ 3}})所以你可以使用Task.Run
:
var myTask1 = Task.Run(() => WorkMethodAsync());
var myTask2 = Task.Run(() => WorkMethodAsync());