3

我有一个 IIS 托管的 WCF 网络服务。

它上面有一个方法(我们称之为“ConfirmOrder”)。调用此方法时,我想 1. 对数据库执行一些快速操作,从而生成 OrderId 2. 启动一个新线程,该线程将执行一些缓慢的工作(例如生成一封电子邮件并发送它) 3. 从1. 同步到客户端。4. 最终,当它完成时,在 2. 中创建的新线程将完成所有其余的处理并发送电子邮件。

问题:

(1) 我确实有如下代码:

// do printing and other tasks 
OrderConfirmedThreadHelper helper = new OrderConfirmedThreadHelper(userSession, result);
// some things first (like generating barcodes) in this thread 
Logger.Write(basket.SessionId, String.Format("Before ConfirmOrderSync"), LogCategoryEnum.Sales, System.Diagnostics.TraceEventType.Verbose);
helper.ConfirmOrderSync();
Logger.Write(basket.SessionId, String.Format("After ConfirmOrderSync"), LogCategoryEnum.Sales, System.Diagnostics.TraceEventType.Verbose);
// slower things (like rendering, sending email) in a separate thread
Thread helperThread = new Thread(new ThreadStart(helper.ConfirmOrderAsync));
helperThread.Start();
return result;

但这似乎引起了问题;至少,该服务一直处于锁定状态。这是一件坏事吗?

(2)我尝试将其更改为

// slower things (like rendering, sending email) in a separate thread            
ThreadPool.QueueUserWorkItem(new WaitCallback(helper.ConfirmOrderAsync));

但是一旦主线程完成,ThreadPool 线程似乎就被杀死了,因为它是一个后台线程。

有没有更好的方法来做到这一点 - 没有编写一个全新的 Windows 服务来与之通信?

4

1 回答 1

0

如果第二个线程在请求​​线程(来自浏览器的线程)之后完成,那么您就有问题了,因为它会被运行时回收并终止。如果您可以等待(如果它只是要发送一封电子邮件,我将等待几秒钟)您可以使用 ManualResetEvent 同步一个线程以等待另一个线程完成并优雅地清理。

如果您等不及了,那么在这种情况下,邮件处理的最佳选择是以下之一

  1. Windows 服务。
  2. 您可以使用 jquery ajax 调用从客户端代码调用一个 .ashx,传递必要的数据以发送邮件。
  3. 一个批处理作业(计划任务、sql server 作业等),它读取要从数据库发送的待处理邮件并发送它们。它会每 X 分钟运行一次,因此您不必担心

希望有帮助!

于 2012-09-17T18:57:14.533 回答