无法处理ThreadAbortException

本文关键字:ThreadAbortException 处理 | 更新日期: 2023-09-27 18:02:03

我有一个WCF服务,它接受请求,并为每个请求调用HTTPWebRequest并返回响应。我使用BlockingCollection来存储传入的请求,并使用一个单独的线程处理集合中的项(使webrequest)。有时Webrequest返回threadabortexception。我捕获它并执行Thread.ResetAbort。但是异常向上流动并且它清除了BlockingCollection。我添加了下面的代码片段。我需要找到一种方法为foreach循环保持继续,即使我得到一个threadabort异常。

public static class YBProcessor
{
    static PriorityQueue<QueuePriorityLevel, string> queue = new PriorityQueue<QueuePriorityLevel, string>();
    static BlockingCollection<KeyValuePair<QueuePriorityLevel, string>> requests;
    static YBProcessor()
    {        
        requests = new BlockingCollection<KeyValuePair<QueuePriorityLevel, string>>(queue);
        Task.Factory.StartNew(() => SendRequestToYB());
    }
    public static void AddCalcRequest(string cusip, double price, QueuePriorityLevel priority)
    {
         requests.Add(new KeyValuePair<QueuePriorityLevel, string>(priority, cusip + "-" + price.ToString()));                                          
    }
   private static void SendRequestToYB()
   {
       // this is a separate thread that processes the requests as the come in.
      foreach (var obj in requests.GetConsumingEnumerable())
      {                 
          try
          {
            var request = GetXML(obj.Value);
            var response = YBClient.GetResponse(request);
            //code to handle response
          }
          catch (ThreadAbortException ex)
          {
            Thread.ResetAbort();
          }
          catch (Exception ex)
          {
          }
       }    
     }
}

//在YBClient的GetResponse方法(只是关键部分。代码无法编译)

private static String GetResponse(String text)
{
    for (iTry = 0; iTry < MAX_TRY; iTry++)
    {
        try
        {
            // Create and setup request
            bytes = encoding.GetBytes(text);
            request = (HttpWebRequest)WebRequest.Create(uri);
            request.Method = "POST";
            request.ContentType = "text/xml";
            request.ContentLength = bytes.Length;
            request.CookieContainer = cookieJar;
            request.Timeout = 100 * 1000;
            request.ReadWriteTimeout = 100 * 1000;
            // Prepare and send data
            postStream = request.GetRequestStream();
            postStream.Write(bytes, 0, bytes.Length);
            postStream.Close();
            // Get response from server
            response = (HttpWebResponse)request.GetResponse();
            if (response.StatusCode == HttpStatusCode.OK)
            {
                reader = new StreamReader(response.GetResponseStream(), encoding);
                xmlResponse = reader.ReadToEnd();
                reader.Close();
                break;
            }               
            response.Close();
        }
        catch (ThreadAbortException ex)
        {
            Thread.ResetAbort();                
            break;              
        }           
        catch (Exception ex)
        {               
            if (ex.GetBaseException() is ThreadAbortException)
            {                   
                Thread.ResetAbort();
                break;
            }
        }
    }       
}
return xmlResponse;
}

无法处理ThreadAbortException

如果服务本身正在调用线程中止异常,那么它可能意味着服务正在停机,在这种情况下,你的线程无法继续工作(想象一个应用程序池回收),如果你想确保你的挂起请求不会丢失,那么你可以做以下其中一个:

1)启动一个新的appdomain http://msdn.microsoft.com/en-us/library/ms173139(v=vs.90).aspx

此方法将减轻服务关闭,但仍不能解决确保所有请求都得到处理的基本问题,因为"其他"应用程序域也可能关闭。

2)更好的解决方案是继续在中央DB或文件中以序列化的形式编写您的请求,并让工作人员不断弹出相同的项目,如果你想简单,为每个请求创建单独的文件,并在处理后删除它们(假设你不会得到数千个请求/秒),对于一个更具可扩展性的解决方案,你可以使用Redis数据库(http://redis.io/)并使用它的"列表"(队列)功能。

注:您可能希望将线程(任务)标记为长时间运行,如果不这样做,它将使用线程池,这对于长时间运行的任务是不推荐的。

 Task.Factory.StartNew(Run, TaskCreationOptions.LongRunning);