我有一个接受请求的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;
}
答案 0 :(得分:1)
如果服务本身正在调用线程中止异常,则可能意味着服务正在关闭,在这种情况下,您的线程无法继续运行(想象一下应用程序池回收),如果您想确保您的待处理请求是没有丢失,那么你可以做以下之一:
1)开始新的appdomain http://msdn.microsoft.com/en-us/library/ms173139(v=vs.90).aspx
此方法可以缓解服务关闭,但仍无法解决确保处理所有请求的根本问题,因为“其他”应用域也可能会停止运行。
2)更好的解决方案是将您的请求以序列化形式写入中央数据库或文件中,并让工作人员不断弹出项目,如果您想要简单,请为每个请求创建单独的文件,处理后删除它们(假设您不会获得数千个请求/秒),对于更具伸缩性的解决方案,您可以使用Redis数据库(http://redis.io/)并使用其“列表”(队列)功能。
P.S。您可能希望将线程(任务)标记为长时间运行,否则,它会使用不建议用于长时间运行任务的线程池。
Task.Factory.StartNew(Run, TaskCreationOptions.LongRunning);