异步操作方法和IO完成端口

vto*_*ola 4 asp.net multithreading task-parallel-library async-await io-completion-ports

当我们的应用程序依赖外部服务时,使用异步编程很重要的原因之一是允许ASP.NET使用IO完成端口,而不是阻止等待外部服务响应的线程,ASP.NET可以将执行停放在IO完成端口并使用该线程来访问另一个请求,只要外部服务响应,然后ASP.NET再次获取该执行并恢复它.这样,没有线程被阻塞.

异步方法的一个例子是:

[HttpPost]
public async Task<ActionResult> Open(String key)
{
    Foo foo= await _externalService.GetFoo(key);
    return View(foo);
}
Run Code Online (Sandbox Code Playgroud)

但是,如果我们对外部服务使用多个请求会发生什么?ASP.NET如何处理它?

[HttpPost]
public async Task<ActionResult> Open()
{
    List<Task<Foo>> tasks = new List<Task<Foo>>();

    foreach (var key in this.Request.Form.AllKeys)
        tasks.Add(_externalService.GetFoo(key));

    var foos = await Task.WhenAll(tasks);

    Foo foo = null;
    foreach (var f in foos)
    {
        if (foo == null && f != null)
            foo = f;
        else
            foo.Merge(f);
    }
    return View(foo);
}
Run Code Online (Sandbox Code Playgroud)

它还在使用IO完成端口吗?或者因为Task.WhenAll阻塞线程?

Ste*_*ary 6

它仍然使用I/O完成端口.WhenAll是异步的,不会阻塞线程.

  • `WhenAll`只是一个带有计数器的`TaskCompletionSource`.当您调用它时,它会将计数器设置为任务数,并为每个递减计数器的任务附加一个延续.当计数器达到零时,TCS完成,这允许您的方法继续.没有线程被阻止. (4认同)