我有一个异步asp.net控制器.该控制器调用异步方法.实际执行异步IO工作的方法深入到我的应用程序中.控制器和链中最后一个方法之间的一系列方法都用async修饰符标记.以下是我如何设置代码的示例:
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工作?
解决方法
是.有一个惩罚(虽然不是一个巨大的),如果你不需要异步不要.这种模式通常称为“返回等待”,您几乎总能删除异步和等待.只需返回代表异步操作的已有任务:
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已在同步调用中运行并完成.