使用Wait()时StreamSocket.InputStreamOptions.ReadAsync挂起

时间:2013-05-08 09:15:18

标签: c# sockets windows-store-apps

这是最小的可能情景,我能够做好准备:

  • 此代码连接到imap.gmail.com
  • 读取初始服务器问候语(使用Read方法)
  • 发送NOOP命令(无操作)
  • 读取NOOP命令响应(再次使用Read方法)

问题是第二次读取挂起。如果使用'await ReadAsync',它可以很好地工作。

当我打破程序时,我可以看到调用堆栈从task.Wait()开始,并以System.Threading.Monitor.Wait()结束。

如果我一步一步地调试它,它就不会挂起。我必须承认它看起来像一个.NET框架错误,但也许我错过了一些明显的东西。

private static async Task<byte[]> ReadAsync(StreamSocket socket)
{
    // all responses are smaller that 1024
    IBuffer buffer = new byte[1024].AsBuffer();
    await socket.InputStream.ReadAsync(
            buffer, buffer.Capacity, InputStreamOptions.Partial);
    return buffer.ToArray();
}

private static byte[] Read(StreamSocket socket)
{
    Task<byte[]> task = ReadAsync(socket);
    task.Wait();
    return task.Result;
}

private async void Button_Click(object sender, RoutedEventArgs e)
{
    Encoding encoding = Encoding.GetEncoding("Windows-1250");

    using (StreamSocket socket = new StreamSocket())
    {
        await socket.ConnectAsync(
            new HostName("imap.gmail.com"), "993", SocketProtectionLevel.Ssl);

        // notice we are using Wait() version here without any problems:
        byte[] serverGreetings = Read(socket);              

        await socket.OutputStream.WriteAsync(
            encoding.GetBytes("A0001 NOOP\r\n").AsBuffer());
        await socket.OutputStream.FlushAsync();

        //byte[] noopResponse = await ReadAsync(socket);    // works
        byte[] noopResponse = Read(socket);                 // hangs
    }
}

1 个答案:

答案 0 :(得分:4)

我解释了这个死锁的原因in my recent MSDN article以及on my blog

简短的回答是,您假设Wait代码上致电Resultasync;你应该一直使用await