我正在努力将线程包含在我的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提供的文档)。等待你已经知道已经完成的线程似乎有点奇怪吗?我能想象的唯一原因是冒泡并处理错误。我在这里错过了什么吗?
答案 0 :(得分:4)
Task.WaitAll()
将抛出AggregateException
- 或者在执行至少一个Task实例期间抛出异常。
ContinueWhenAll()
不会抛出此异常,它只是在所有内容完成取消后启动您的上一个任务等。
答案 1 :(得分:2)
首先,我注意到你使用的List<T>
与Parallel.ForEach
不是线程安全的,你应该用并发集合替换它,例如:ConcurrentQueue<T>
。
关于WaitAll
vs ContinueWhenAll
,如果任何任务出现故障,WaitAll
将抛出,因此上面的代码是验证所有任务已成功完成,您也可以这样做如果您将ContinuationOptions
参数传递给ContinueWhenAll
,例如OnlyRanToCompeletion
,那么仅当所有任务都成功完成时才会安排继续任务。