3
public class HomeController : Controller
{
    public ActionResult Index()
    {
        Task<string> t = DownloadStringTaskAsync(new Uri("http://www.google.com/search?q=aspnet"));
        t.Wait();
        string res = t.Result;

        return View((object)res);
    }

    Task<string> DownloadStringTaskAsync(Uri address)
    {
        TaskCompletionSource<string> tcs = new TaskCompletionSource<string>();
        WebClient wc = new WebClient();
        wc.DownloadStringCompleted += (s, e) => tcs.SetResult(e.Result);
        wc.DownloadStringAsync(address);
        return tcs.Task;
    }
}

该操作将永远不会返回,请求将简单地超时。

我们将 TPL 任务用​​于所有异步操作,并且必须包装一些 WebClient 的方法,直到我们切换到框架 4.5。

上述方法DownloadStringTaskAsync是其中一个包装器的简化版本。

我无法弄清楚为什么等待任务会使 aspnet 中的线程死锁并且在控制台或桌面应用程序中工作正常。

这是死锁线程的样子。

请求线程

[处于睡眠、等待或加入]
mscorlib.dll!System.Threading.Monitor.Wait(object obj, int millisecondsTimeout, bool exitContext) + 0x18 bytes mscorlib.dll!System.Threading.ManualResetEventSlim.Wait(int millisecondsTimeout, System .Threading.CancellationToken cancelToken) + 0x264 字节
mscorlib.dll!System.Threading.Tasks.Task.InternalWait(int millisecondsTimeout, System.Threading.CancellationToken cancelToken) + 0x166 字节 mscorlib.dll!System.Threading.Tasks.Task.Wait( int millisecondsTimeout, System.Threading.CancellationToken cancelToken) + 0x41 字节
mscorlib.dll!System.Threading.Tasks.Task.Wait() + 0xb 字节
TCSProblem.DLL!TCSProblem.Controllers.HomeController.Index() 第 16 行 + 0xa 字节

WebClient 线程

[在睡眠、等待或加入]
mscorlib.dll!System.Threading.Monitor.Enter(object obj, ref bool lockTaken) + 0x14 字节
System.Web.dll!System.Web.AspNetSynchronizationContext.CallCallback(System.Threading. SendOrPostCallback 回调,对象状态) + 0x53 字节
System.Web.dll!System.Web.AspNetSynchronizationContext.Post(System.Threading.SendOrPostCallback 回调,对象状态) + 0x9 字节
System.dll!System.ComponentModel.AsyncOperation.Post(System. Threading.SendOrPostCallback d,对象 arg) + 0x29 字节
System.dll!System.Net.WebClient.PostProgressChanged(System.ComponentModel.AsyncOperation asyncOp, System.Net.WebClient.ProgressData 进度) + 0x25c 字节

4

2 回答 2

11

不幸的是,WebClient 捕获了当前的同步上下文并通过发布到它来引发完成事件。因此,完成操作仅在您从您的操作方法返回后运行,这永远不会发生。这意味着,你要么...

  1. ...需要将您对 DownloadStringAsync 的调用包装在 Task.Factory.StartNew 中,以便切换到没有同步上下文的线程
  2. ...需要在调用前将 SynchronizationContext.Current 设置为 null,然后再恢复
  3. ...使用异步 MVC 操作
于 2012-06-28T11:33:49.657 回答
-1

默认情况下,所有 .NET 类都捕获 SynchronizationContext,我遇到了类似的问题:SmtpClient.SendAsync blocking my ASP.NET MVC Request

于 2012-06-28T11:37:44.810 回答