我正在编写一个仅使用一个套接字的简单套接字客户端。所以我想我可以为所有接收操作重用一个 SocketAsyncEventArgs。只要连接了 socket ,它就可以正常工作。当服务器断开连接时,客户端会在控制台中进入一个无限接收循环垃圾邮件“接收成功”。

当套接字断开连接时,e.SocketError != SocketError.Success 不应该是真的吗?

这是代码的一部分:

private void Completed(object sender, SocketAsyncEventArgs e)
{
   if (e.SocketError != SocketError.Success)
      status = 0;

   System.Console.WriteLine(e.LastOperation + " " + e.SocketError);

   if (status == 1)
   {
      switch (e.LastOperation)
      {
         case SocketAsyncOperation.Connect:
            ProcessConnect(e);
            break;
         case SocketAsyncOperation.Receive:
            ProcessReceive(e);
            break;
         case SocketAsyncOperation.Send:
            ProcessSend(e);
            break;
         default:
            status = 0;
            break;
      }
   }

   if (status != 1)
      CloseSocket();
}

private void ProcessReceive(SocketAsyncEventArgs e)
{
   if (!socket.ReceiveAsync(e))
      Completed(null, e);
}

最佳答案

如果您的套接字设置为字节流,那么您可能会收到零字节的回调,这表明套接字已正常关闭并且不会再读取任何字节。只需在您的代码中检查这一点,例如

private void ProcessReceive( SocketAsyncEventArgs e )
{
    AsyncUserToken token = e.UserToken as AsyncUserToken;
    if( e.ByteTransferred > 0 && e.SocketError == SocketError.Success )
    {
        // .. process normally ..
    }
    else
    {
        CloseClientSocket( e );
    }
}

private void CloseClientSocket( SocketAsyncEventArgs e )
{
    AsyncUserToken token = e.UserToken as AsyncUserToken;
    try
    {
        token.Socket.Shutdown( SocketShutdown.Send );
    }
    // throws error if it's already closed
    catch( Exception ) {}
    token.Socket.Close();
}

当然,您还需要阻止客户端尝试使用套接字,但我相信您可以解决这个问题。

关于c# - 将 SocketAsyncEventArgs 与 ReceiveAsync 重用会导致无限循环,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/1366427/

10-11 23:17
查看更多