18

我遇到了特定于 Windows 8 和 VS2012 的问题。

我有一个 TCP 套接字服务器和客户端,并且正在本地网络上进行一些测试。使用 sysinternals TCPView,我可以看到数据包从 TCP 客户端发送并到达 TCP 服务器(我看到数据包计数器增加)。

但是,看起来好像数据没有进入应用程序堆栈?同样的构建在 Windows 7 上运行没有问题。

我关闭了 Windows 8 防火墙,并在关闭 UAC 的域管理员用户上以提升的权限运行这两个进程。

当我将客户端连接到外部服务器(在单独的机器上运行)时,一切正常。Windows 8 中是否还有其他内容可以禁止本地进程之间的 TCP 数据通信?

谢谢,

编辑

为了确保我的服务器应用程序中没有任何东西导致此问题,我在控制台应用程序中构建了一个快速 TCP 服务器,并为套接字构造函数使用以下代码:

listenerSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);

并侦听与我的服务器应用程序相同的本地 IP/端口。我遇到了同样的问题,我可以远程登录到端口,但 listenerSocket.AcceptAsync 永远不会被击中。

编辑 2

经过进一步测试,我发现我的问题与使用异步套接字调用有关,即如果我使用像 socket.Accept() 这样的同步调用,则测试应用程序运行正常。但是,当我使用异步套接字调用时,即 socket.AcceptAsync(),我遇到了上述问题。到目前为止,在异步套接字调用方面,我找不到任何关于 win7 和 8 之间差异的提及。

这是我的快速示例应用程序,它显示异步回调从未被触发。此代码段在 Windows 7 中运行良好,但在 Windows 8 中无法运行(尝试 telnet 到 127.0.0.1 : 7000)。

    class Program
{
    private static SocketAsyncEventArgs socketAsyncEventArgs = new SocketAsyncEventArgs();

    static void Main(string[] args)
    {
        var listenerSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
        listenerSocket.Bind(new IPEndPoint(IPAddress.Parse("127.0.0.1"), 7000));
        listenerSocket.Listen(100);

        socketAsyncEventArgs.Completed += AcceptEventArg_Completed;
        listenerSocket.AcceptAsync(socketAsyncEventArgs);

        Console.ReadLine();
    }

    private static void AcceptEventArg_Completed(object sender, SocketAsyncEventArgs e)
    {
        Console.WriteLine("AcceptEventArg_Completed");
    }
}

编辑 3

我发现另外 2 个人在 Microsoft Connect 上报告了同样的问题: https ://connect.microsoft.com/VisualStudio/feedback/details/759913/socketasynceventargs-completed-doesnt-fire-in-net-framework-4-5 和 http: //connect.microsoft.com/VisualStudio/feedback/details/747218/saea-not-working-in-net-4-5-rp

而第二个很有趣,因为它似乎得出结论,在 Console.ReadLine() 调用中有一个 Windows 错误,它导致了问题并阻止了异步回调。如果我将代码段中的 Console.ReadLine() 替换为:

            while (true)
        {
            System.Threading.Thread.Sleep(10);
        }

一切正常。

4

3 回答 3

4

看到这个:GetQueuedCompletionStatus can't dequeue IO from IOCP if the thread that original issue is blocking in ReadFile under windows 8

这是 Windows 8 和 2012 中的一个错误,会影响所有使用 AcceptEx 和 ReadFile 的程序。目前,已知只有这两个功能受到影响。

于 2012-09-28T07:47:37.250 回答
0

我在使用 SocketAsyncEventArgs 开发 Tcp 服务器和客户端应用程序时遇到了同样的事情

我建议你先试试这个。

  1. 打开具有高级安全性的 Windows 防火墙检查入站/出站规则以查看您的应用程序是否被阻止。

  2. 打开 AssemblyInfo.cs 并更改

[程序集:Guid("06985fe3-80eb-48b4-940a-fd926e2f2053")]

到任何其他指导值。

通过更改此设置,Windows 会认为这是一个新应用程序,如果对旧应用程序有任何限制,它将不会出现在新应用程序上。

于 2012-09-19T14:05:55.737 回答
0

听起来这个 Windows 错误与 Windows 8 上的 IOCP 处理(可能只是 AcceptEx)有关,而其他阻塞 I/O 正在同一线程上进行:

http://social.technet.microsoft.com/Forums/en-GB/winserver8gen/thread/5764cd0f-fda1-4cfa-ae35-808210bae77e

所以套接字连接被接受,但你的应用程序永远不会收到它的通知。

也许 Windows 8 做了一些奇怪的、稍微损坏的巫术来将像 Console.Read 这样的同步 IO 在内部转换为异步。

您可以将服务器代码移动到不同的线程中,其他尝试的解决方法可能是同步执行 Accept 或将控制台处理更改为异步(我无法真正尝试,因为我没有 Windows 8)。

于 2012-09-19T15:16:49.800 回答