3

我正在做一个 Nancy/ASP.Net 项目。我尝试使用 WebRequest 从其他 Web 服务获取数据,并在异步函数中实现了请求。我发现一个问题,当我尝试等待从函数返回的任务时,它会被无限期地阻塞。简化的代码看起来像这样..

using System.Net;
using System.Threading.Tasks;
using Nancy;

public class TestModule : NancyModule{
    public TestModule() {
        Get["/"] = p => "Hello, world";
        Get["/async/"] = p => {
            var req = WebRequest.CreateHttp("http://localhost:13254/");

//          var responseTask = req.GetResponseAsync();  // this works!
            var responseTask = getResponse(req);   // this gets blocked!
            var waitSuccess = responseTask.Wait(10000);

            return waitSuccess ? "Yeah!" : "woooh!";
        };
    }
    async Task<HttpWebResponse> getResponse(HttpWebRequest request) {
        return (HttpWebResponse) await request.GetResponseAsync();
    }
}

该代码使用 NancyFx,但它也发生在 vanilla ASP.Net 页面上。localhost:13254 上的服务运行良好。如果我使用从请求的 GetResponseAsync() 直接返回的任务,代码可以正常工作,但如果我将它包装在异步方法中,它就会被阻塞。

有谁知道代码有什么问题?:(我可以更改为使用同步版本,但异步功能可以在其他自托管服务中找到......所以如果可能的话,我也想在这里使用相同的代码......

4

2 回答 2

4

我在我的博客最近的 MSDN 文章中描述了这种死锁行为。

要解决此问题,您可以在任何ConfigureAwait(false)地方使用,也可以使用同步方法。理想的解决方案是一直使用await,永远不要使用Waitor Result,但这在您的情况下可能是不可能的(只有当 Nancy 与async代表一起工作时,它才有效,即Get["/async/"] = async p => { ... };)。

于 2013-06-08T16:56:18.197 回答
1

除了 Stephen 提供的解决方案之外,我还通过在 Nancy 路由处理程序的根处更改同步上下文来解决。给定这个效用函数:

using System;
using System.Threading;

namespace Utility.Async{
    static public class TPContext{
        static public T Call<T>(Func<T> handler) {
            var currentContext = SynchronizationContext.Current;
            SynchronizationContext.SetSynchronizationContext(null);
            try {
                return handler();
            }finally {
                SynchronizationContext.SetSynchronizationContext(currentContext);
            }
        } 
    }
}

只需使用此函数包装处理程序

        Get["/async/"] = p => TPContext.Call(() => {
            var req = WebRequest.CreateHttp("http://localhost:13254/");

            var responseTask = getResponse(req);
            var waitSuccess = responseTask.Wait(10000);

            return waitSuccess ? "Yeah!" : "woooh!";
        });

它只是工作 ^^a 我不认为我需要 ASP.Net 上下文.. 实际上我希望 Nancy 会创建自己的同步上下文,因为 Nancy 在概念上与 Asp.net 的工作方式不同。

于 2013-06-09T05:11:24.687 回答