【问题标题】:WCF client blocks on async methodsWCF 客户端阻止异步方法
【发布时间】:2013-11-21 04:45:09
【问题描述】:

我正在开发 WCF 客户端应用程序,并且遇到了等待/异步模式的困难。 似乎该行: 等待客户端.LongOperationAsync(); 总是阻塞。据我了解,线程应该退出并继续到 Main() 方法,然后在异步方法完成时返回,也许我错了。

下面代码的输出是(总是):

测试()开始
测试()错误
*
*
*
...

Test() 方法总是在上下文返回到 main 之前 完成。任何想法将不胜感激。

static void Main(string[] args)
{
    Program p = new Program();
    p.Test();
    while (true)
    {
        Console.WriteLine("*");
        Thread.Sleep(500);
    }       
}
private async Task Test()
{
    Console.WriteLine("Test() started");
    try
    {
        MySoapClient client = new MySoapClient(
            new BasicHttpBinding(new BasicHttpSecurityMode()),
            new EndpointAddress("http://badaddress"));
        await client.LongOperationAsync();
        Console.WriteLine("Test() success");
    }
    catch (Exception)
    {
        Console.WriteLine("Test() error");
        return;
    }
    Console.WriteLine("Test() end successfully");
}

【问题讨论】:

  • 请记住,在这种情况下,您不应依赖于代码执行的顺序。由于您不是在等待 Test 方法完成,因此您无法保证它是在调用方法中的任何后续代码行之前还是之后完成。
  • 因此,按照它完成的顺序完成不会被视为错误。

标签: c# .net wcf asynchronous async-await


【解决方案1】:

异步方法同步执行,直到第一个await;如果您的LongOperationAsync 方法在第一次等待之前执行了阻塞操作,则调用方法也将被阻塞。我怀疑这就是你的情况。

这可能是因为WebRequest.BeginGetResponse 同步执行它的一些工作。请参阅 Stephen Toub 对this question 的回答:

Async CTP 的 GetRequestStreamAsync 和 GetResponseAsync 很简单 围绕现有的 HttpWebRequest.BeginGetRequestStream 和 .NET 4 中的 BeginGetResponse。这些 Begin* 方法有很多设置 他们在他们之前所做的工作(例如代理、DNS、连接池等) 可以提交请求,不幸的是今天所有的工作都发生了 作为 Begin* 调用的一部分同步进行。

在这种情况下,您提供了一个错误的域名,所以我怀疑 DNS 解析失败需要一段时间。

【讨论】:

  • 感谢您回复托马斯。我认为情况确实如此。这似乎是 WCF 代理的一个非常糟糕的行为。这几乎使得从 UI 应用程序直接使用 WCF 代理变得不可能,对吧?是否还有其他具有“真正”异步 API 的代理?
  • @AdamLevi,好吧,这不是不可能的,但如果你想确保它永远不会阻塞,你必须在一个单独的线程中进行......但我同意,这很糟糕。 AFAIK,所有代理实现最终都会调用WebRequest.BeginGetResponse,所以他们都会遇到同样的问题。但是,您可能可以使用 Dns 类中的异步方法预先检测到域不存在。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2012-10-03
  • 1970-01-01
  • 2012-07-12
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多