我有一个异步“上游”的方法。我正在尝试遵循最佳实践并在堆栈中一直进行qith异步。
在MVC中的Controller操作中,我可以预见到死锁问题如果我依赖.Result()。
将Controller操作更改为async似乎是要走的路,尽管问题是在lambda中多次调用async方法。
如何等待返回多个结果的lamda ?
public async Task<JsonResult> GetLotsOfStuff()
{
IEnumerable<ThingDetail> things= previouslyInitialisedCollection
.Select(async q => await GetDetailAboutTheThing(q.Id)));
return Json(result, JsonRequestBehavior.AllowGet);
}
你可以看到我尝试使lambda异步,但这只是一个编译器异常:
无法转换来源类型
System.Collections.Generic.IEnumerable<System.Threading.Tasks.Task<ThingDetail>
定位类型System.Collections.Generic.IEnumerable<ThingDetail>
我在哪里错了?
答案 0 :(得分:15)
Thing
集合转换为Task<Thing>
s。Task.WhenAll
加入所有这些任务并等待它。Thing[]
public async Task<JsonResult> GetLotsOfStuff()
{
IEnumerable<Task<ThingDetail>> tasks = collection.Select(q => GetDetailAboutTheThing(q.Id));
Task<int[]> jointTask = Task.WhenAll(tasks);
IEnumerable<ThingDetail> things = await jointTask;
return Json(things, JsonRequestBehavior.AllowGet);
}
或者,简洁地使用类型推断:
public async Task<JsonResult> GetLotsOfStuff()
{
var tasks = collection.Select(q => GetDetailAboutTheThing(q.Id));
var things = await Task.WhenAll(tasks);
return Json(things, JsonRequestBehavior.AllowGet);
}
小提琴:https://dotnetfiddle.net/78ApTI
注意:由于GetDetailAboutTheThing
似乎返回Task<Thing>
,因此惯例是将Async
附加到其名称 - GetDetailAboutTheThingAsync
。