在我的Windows服务中,我创建了一个“父”前台线程,该线程反过来使用ThreadPool(这意味着它们是后台)生成“子”线程来执行任务。
在Windows服务停止时优雅地关闭前台线程的最佳方法是什么?
这是我当前的实现(从特定于任务的逻辑中删除):
public partial class TaskScheduler : ServiceBase
{
private static AutoResetEvent _finishedTaskAutoResetEvent = new AutoResetEvent(false);
//This flag is used to increase chances of the Spawning Thread to finish gracefully when service stops.
private bool StopRequested { get; set; }
private int _executingTasksCount;
private int ExecutingTasksCount { get { return _executingTasksCount; } }
private void IncCurrentTasksCount()
{
Interlocked.Increment(ref _executingTasksCount);
}
private void DecCurrentTasksCount()
{
Interlocked.Decrement(ref _executingTasksCount);
}
public TaskScheduler()
{
InitializeComponent();
Thread spawningThread = new Thread(DoSpawnTaskExecutionThreads);
spawningThread.Name = "Spawning Thread";
spawningThread.IsBackground = false;
spawningThread.Start();
}
protected override void OnStart(string[] args)
{
}
protected override void OnStop()
{
StopRequested = true;
}
private void DoSpawnTaskExecutionThreads()
{
//We check StopRequested to try and finish this thread gracefully when service stops.
while (!StopRequested)
{
while (!StopRequested && ExecutingTasksCount < MaxPooledTasks)
{
ThreadPool.QueueUserWorkItem(ExecuteTask, new Task());
IncCurrentTasksCount();
}
_finishedTaskAutoResetEvent.WaitOne();
}
//Either all task execution threads will finish or the process will be terminated forcibly.
while (ExecutingTasksCount > 0)
{
Thread.Sleep(200); //Check five times a second.
}
_eventLog.WriteEntry("The Spawning Thread finished along with task execution threads.");
}
private void ExecuteTask(object state)
{
try
{
Task task = (Task)state;
task.Execute();
}
catch
{
// Handle exception.
}
finally
{
DecCurrentTasksCount();
_finishedTaskAutoResetEvent.Set();
}
}
}
答案 0 :(得分:3)
我发现代码存在一些问题。
StopRequested
的检查不是线程安全的。ExecutingTaskCount
的检查不是线程安全的。_finishedTaskAutoResetEvent
AutoResetEvent
信号可能会丢失,因为WaitHandle
不会保持计数。也许这就是你想要的,但它可能导致嵌套while
循环的一些奇怪旋转。以下是重构代码的方法。它使用.NET 4.0中提供的CountdownEvent
类。
public class TaskScheduler : ServiceBase
{
private m_Stop as ManualResetEvent = new ManualResetEvent(false);
protected override void OnStart(string[] args)
{
var thread = new Thread(DoSpawnTaskExecutionThreads);
thread.Name = "Spawning Thread";
thread.IsBackground = false;
thread.Start();
}
protected override OnStop()
{
m_Stop.Set();
}
public DoSpawnTaskExecutionThreads()
{
// The semaphore will control how many concurrent tasks can run.
var pool = new Semaphore(MaxPooledThreads, MaxPooledThreads);
// The countdown event will be used to wait for any pending tasks.
// Initialize the count to 1 so that we treat this thread as if it
// were a work item. This is necessary to avoid a subtle race
// with a real work item that completes quickly.
var tasks = new CountdownEvent(1);
// This array will be used to control the spinning of the loop.
var all = new WaitHandle[] { pool, m_Stop };
while (WaitHandle.WaitAny(all) == 0)
{
// Indicate that there is another task.
tasks.AddCount();
// Queue the task.
Thread.QueueUserWorkItem(
(state) =>
{
try
{
var task = (Task)state;
task.Execute();
}
finally
{
pool.Release(); // Allow another task to be queued.
tasks.Signal(); // Indicate that this task is complete.
}
}, new Task());
}
// Indicate that the main thread is complete.
tasks.Signal();
// Wait for all pending tasks.
tasks.Wait();
}
}
答案 1 :(得分:2)
我在这里看到一个问题:
StopRequested不应该是自动属性。您应该将其定义为具有支持字段的属性,以便将其标记为volatile。
private volatile bool stopRequested;
private bool StopRequested
{
get { return this.stopRequested; }
set { this.stopRequested = value; }
}
如果没有这个,当服务设置时,你的线程可能不会(至少是马上就会)看到退出条件。
此外,如果选择.NET 4,可以使用CancellationToken
和BlockingCollection<T>
进行更简单的设计。
答案 2 :(得分:0)
您可以使用Join方法“优雅地”杀死该线程。 MSDN有一些关于该方法的信息。