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