我有一个异步asp.net控制器.该控制器调用异步方法.实际执行异步IO工作的方法深入到我的应用程序中.控制器和链中最后一个方法之间的一系列方法都用async修饰符标记.以下是我如何设置代码的示例:
@H_404_2@public async Task<ActionResult> Index(int[] ids)
{
List<int> listOfDataPoints = dataPointService(ids);
List<Task> dpTaskList = new List<Task>();
foreach (var x in listOfDataPoints)
{
dpTaskList.Add(C_Async(x));
}
await Task.WhenAll(dpTaskList);
return View();
}
private async Task C_Async(int id)
{
//this method executes very fast
var idTemp = paddID(id);
await D_Async(idTemp);
}
private async Task D_Async(string id)
{
//this method executes very fast
await E_Async(id);
}
private async Task E_Async(string url)
{
//this method performs the actual async IO
result = await new WebClient().DownloadStringTaskAsync(new Uri(url))
saveContent(result);
}
正如您所看到的,控制器异步调用C_Async(x),然后有一系列异步方法到E_Async.控制器和E_Async之间有方法,并且都有async修饰符.是否存在性能损失,因为有使用异步修改器的方法但没有执行任何异步IO工作?
解决方法
是.有一个惩罚(虽然不是一个巨大的),如果你不需要异步不要.这种模式通常称为“返回等待”,您几乎总能删除异步和等待.只需返回代表异步操作的已有任务:
@H_404_2@private Task C_Async(int id)
{
// This method executes very fast
var idTemp = paddID(id);
return D_Async(idTemp);
}
private Task D_Async(string id)
{
// This method executes very fast
return E_Async(id);
}
在此特定情况下,Index将仅等待E_Async返回的任务.这意味着在完成所有I / O之后,下一行代码将直接返回View();. C_Async和D_Async已在同步调用中运行并完成.