Task.Wait in ContinueWhenAll Action

时间:2011-08-10 08:10:36

标签: c# azure task plinq

我正在努力将线程包含在我的azure代码中,以便将东西放在队列中。为此,我使用http://www.microsoft.com/download/en/details.aspx?id=19222作为参考。

我将多条消息排入队列的代码如下:

public void AddMessagesAsync(IEnumerable<IQueueMessage> messages, string queue = null, TimeSpan? timeToLive = null)
{
  //check if we need to switch queues
  if (!String.IsNullOrEmpty(queue))
  {
    SetCurrent(queue);
  }

  //setup list of messages to enqueue
  var tasks = new List<Task>();
  Parallel.ForEach(messages, current => {
    if (timeToLive.HasValue)
    {
      //create task with TPL
      var task = Task.Factory.FromAsync(Current.BeginAddMessage, Current.EndAddMessage, Convert(current), timeToLive.Value, tasks); 
      //setup continuation to trigger eventhandler
      tasks.Add(task.ContinueWith((t) => AddMessageCompleted(t)));
    }
    else
    {
      //create task with TPL
      var task = Task.Factory.FromAsync(Current.BeginAddMessage, Current.EndAddMessage, Convert(current), tasks);
      //setup continuation to trigger eventhandler
      tasks.Add(task.ContinueWith((t) => AddMessageCompleted(t)));
    }
  });

  //setup handler to trigger when all messages are enqueued, a we are blocking the thread over there to wait for all the threads to complete
  Task.Factory.ContinueWhenAll(tasks.ToArray(), (t) => AddMessagesCompleted(t));               
}

private void AddMessagesCompleted(Task[] tasks)
{
  try
  {
    //wait for all tasks to complete
    Task.WaitAll(tasks);
  }
  catch (AggregateException e)
  {
    //log the exception
    var ex = e;
    //return ex;
  }

  if (AddedMessages != null)
  {
    AddedMessages(tasks, EventArgs.Empty);
  }
}

现在我的问题是关于继续中的Task.Wait(根据MS提供的文档)。等待你已经知道已经完成的线程似乎有点奇怪吗?我能想象的唯一原因是冒泡并处理错误。我在这里错过了什么吗?

2 个答案:

答案 0 :(得分:4)

当至少有一个Task实例被取消时,

Task.WaitAll()将抛出AggregateException - 或者在执行至少一个Task实例期间抛出异常。

ContinueWhenAll()不会抛出此异常,它只是在所有内容完成取消后启动您的上一个任务等。

答案 1 :(得分:2)

首先,我注意到你使用的List<T>Parallel.ForEach不是线程安全的,你应该用并发集合替换它,例如:ConcurrentQueue<T>

关于WaitAll vs ContinueWhenAll,如果任何任务出现故障,WaitAll将抛出,因此上面的代码是验证所有任务已成功完成,您也可以这样做如果您将ContinuationOptions参数传递给ContinueWhenAll,例如OnlyRanToCompeletion,那么仅当所有任务都成功完成时才会安排继续任务。