等待QueueUserWorkItem完成

时间:2011-06-30 03:29:01

标签: c# .net multithreading threadpool parallel-processing

如果我使用QueueUserWorkItem向线程池添加作业...在完成所有作业之前,如何使程序继续运行?

我知道我可以添加一些逻辑来阻止应用程序运行,直到完成所有作业,但我想知道是否有类似Thread.Join()的内容,或者是否有任何方法可以检索正在分配的每个线程一份工作。

3 个答案:

答案 0 :(得分:42)

您可以使用事件进行同步。像这样:

private static ManualResetEvent resetEvent = new ManualResetEvent(false);

public static void Main()
{
    ThreadPool.QueueUserWorkItem(arg => DoWork());
    resetEvent.WaitOne();
}

public static void DoWork()
{
    Thread.Sleep(5000);
    resetEvent.Set();
}

如果您不想将事件集嵌入到您的方法中,您可以执行以下操作:

var resetEvent = new ManualResetEvent(false);
ThreadPool.QueueUserWorkItem(
    arg => 
    {
        DoWork();
        resetEvent.Set();
    });
resetEvent.WaitOne();

对于多个项目:

var events = new List<ManualResetEvent>();

foreach(var job in jobs)
{   
    var resetEvent = new ManualResetEvent(false);
    ThreadPool.QueueUserWorkItem(
        arg =>
        {
            DoWork(job);
            resetEvent.Set();
        });
    events.Add(resetEvent);
}
WaitHandle.WaitAll(events.ToArray());

答案 1 :(得分:17)

执行此操作的最佳方法是使用CountdownEvent类。这是一个相当完善的模式,并且具有可扩展性。

using (var finished = new CountdownEvent(1))
{
  foreach (var workitem in workitems)
  {
    var capture = workitem; // Used to capture the loop variable in the lambda expression.
    finished.AddCount(); // Indicate that there is another work item.
    ThreadPool.QueueUserWorkItem(
      (state) =>
      {
        try
        {
          ProcessWorkItem(capture);
        }
        finally
        {
          finished.Signal(); // Signal that the work item is complete.
        }
      }, null);
  }
  finished.Signal(); // Signal that queueing is complete.
  finished.Wait(); // Wait for all work items to complete.
}

答案 2 :(得分:3)

您可以使用.NET的Barrier类来实现此目的。

Barrier barrier = new Barrier(3); 
for(int i = 0; i < 2; i++)
{
    ThreadPool.QueueUserWorkItem(
    (state) =>
    {
       foo();
       barrier.SignalAndWait();
    }, null);
}
barrier.SignalAndWait();