52

好的,我的代码抛出了一个奇怪的异常,这一直困扰着我多年。

System.Net.Sockets.SocketException: A blocking operation was interrupted by a call to WSACancelBlockingCall
   at System.Net.Sockets.Socket.Accept()
   at System.Net.Sockets.TcpListener.AcceptTcpClient()

MSDN 对此并没有太大帮助:http: //msdn.microsoft.com/en-us/library/ms741547 (VS.85).aspx我什至不知道如何开始解决这个问题。它每天只抛出 4 或 5 次,而且从不在我们的测试环境中。仅在生产地点和所有生产地点。

我发现很多帖子都在询问这个异常,但没有关于导致它的原因以及如何处理或防止它的实际明确答案。

代码在单独的后台线程中运行,方法开始:

public virtual void Startup()
    {
     TcpListener serverSocket= new TcpListener(new IPEndPoint(bindAddress, port));    
        serverSocket.Start();

然后我运行一个循环,将所有新连接作为作业放在单独的线程池中。由于应用程序架构,它变得更加复杂,但基本上:

   while (( socket = serverSocket.AcceptTcpClient()) !=null) //Funny exception here
    {
         connectionHandler = new ConnectionHandler(socket, mappingStrategy);
         pool.AddJob(connectionHandler);
    }
  }

从那里开始,pool它拥有自己的线程,分别处理自己线程中的每个作业。

我的理解是 AcceptTcpClient() 是一个阻塞调用,并且 winsock 以某种方式告诉线程停止阻塞并继续执行..但是为什么呢?我该怎么办?只是捕获异常并忽略它?


好吧,我确实认为其他一些线程正在关闭套接字,但它肯定不是来自我的代码。我想知道的是:此套接字是由连接客户端(在套接字的另一侧)关闭还是由我的服务器关闭。因为此时此刻,每当发生此异常时,它都会关闭我的侦听端口,从而有效地关闭我的服务。如果这是从远程位置完成的,那么这是一个主要问题。

或者,这是否只是 IIS 服务器关闭了我的应用程序,从而取消了我所有的后台线程和阻塞方法?

4

5 回答 5

52

serverSocket 是否有可能从另一个线程关闭?这将导致此异常。

于 2008-08-29T11:14:51.233 回答
7

这是我避免 WSAcancelblablabla 的示例解决方案:将您的线程定义为全局,然后您可以使用这样的调用方法:

private void closinginvoker(string dummy)
    {
        if (InvokeRequired)
        {
            this.Invoke(new Action<string>(closinginvoker), new object[] { dummy });
            return;
        }
        t_listen.Abort();
        client_flag = true;
        c_idle.Close();
        listener1.Stop();
    }

调用它后,先关闭线程,然后关闭永远循环标志,以便它阻止进一步等待(如果有的话),然后关闭 tcpclient 然后停止侦听器。

于 2011-12-06T08:37:18.237 回答
5

这可能发生在serverSocket.Stop(). 每当Dispose被调用时,我都会调用它。

这是我对监听线程的异常处理的样子:

try
{
    //...
}
catch (SocketException socketEx)
{    
    if (_disposed)
        ar.SetAsCompleted(null, false); //exception because listener stopped (disposed), ignore exception
    else
        ar.SetAsCompleted(socketEx, false);
}

现在发生的事情是,在_disposed设置为 true 之前,每隔一段时间就会发生一次异常。所以对我来说,解决方案是让一切线程安全。

于 2008-11-04T10:00:53.303 回答
3

同样在这里!但我发现,“服务器端”上的 ReceiveBuffer 被客户端淹没了!(在我的情况下,一堆 RFID 扫描仪,他们一直在发送垃圾邮件 TagCode,而不是在下一个 TagCode 到达之前停止发送)

它有助于提高 ReceiveBuffers 并重新配置扫描仪......

于 2011-08-26T11:37:45.180 回答
0

最近,我在使用 HttpWebRequest PUT 一个大文件并且超时时间已过时看到了这个异常。

只要您的上传时间> 3秒,就可以使用以下代码,据我所知,它会导致此错误。

string path = "Reasonably large file.dat";
int bufferSize = 1024;
byte[] buffer = new byte[bufferSize];
System.Net.HttpWebRequest req = (HttpWebRequest)System.Net.HttpWebRequest.Create("Some URL");
req.Method = "PUT";
req.Timeout = 3000; //3 seconds, small timeout to demonstrate
long length = new System.IO.FileInfo(path).Length;
using (FileStream input = File.OpenRead(path))
{
    using (Stream output = req.GetRequestStream())
    {
        long remaining = length;
        int bytesRead = 0;
        while ((bytesRead = input.Read(buffer, 0, (int)Math.Min(remaining, (decimal)bufferSize))) > 0)
        {
            output.Write(buffer, 0, bytesRead);
            remaining -= bytesRead;
        }
        output.Close();
    }
input.Close();
}
于 2018-06-26T03:52:26.643 回答