在Asp.net Core中使用async函数时,它们可能被分配到不同的线程上执行。这可能会导致一些继续函数在错误的线程上执行,从而导致死锁或其他问题。以下是一个示例,说明如何在Asp.net Core应用程序中处理这种情况:
public async Task Index()
{
List values = await GetValueAsync();
return View(values);
}
private Task> GetValueAsync()
{
var task = Task.Run(() => GetData());
return task;
}
private List GetData()
{
// Some long running task
return new List() { "value1", "value2" };
}
在上面的示例中,从GetValueAsync
方法中返回的任务将在一个不同的线程上执行。这使得继续函数在尝试访问UI资源时可能会抛出错误。
解决这个问题的方法是使用ConfigureAwait
方法并将其设置为false
。这告诉async函数不在当前线程上等待延续函数。如果需要在UI线程上运行继续函数,则必须确保它通过调用Invoke
方法在UI线程上调用:
public async Task Index()
{
List values = await GetValueAsync().ConfigureAwait(false);
return View(values);
}
private async Task> GetValueAsync()
{
var data = await Task.Run(() => GetData()).ConfigureAwait(false);
return data;
}
private List GetData()
{
// Some long running task
return new List() { "value1", "value2" };
}
在上面的示例中,ConfigureAwait(false)
告诉async函数,继续函数不必在UI线程中运行。然后,我们可以在异步方法中使用ConfigureAwait(false)
来确保我们不会在错误的线程上等待延续