我正在尝试重新启动多个任务中的一个,如果它失败了。我使用
例如:
.ContinueWith(t => { HandleException(t); }, TaskContinuationOptions.OnlyOnFaulted);
,其中HandleException(t)
方法应在现有任务数组中找到该任务并创建一个新任务来代替它。但是,传递给我的HandleException(t)
方法的任务与原始任务不同,因此我无法在现有任务中找到它。此外,在处理异常时,现有任务仍在运行。例如:
using System;
using System.Threading.Tasks;
using System.Threading;
static Task[] tasks;
static void Main(string[] args)
{
tasks = new Task[2];
for (int i = 0; i < tasks.Count(); i++)
{
tasks[i] = Task.Run(() => { Thread.Sleep(1000); throw new Exception("BOOM"); })
.ContinueWith(t => { HandleException(t); }
, TaskContinuationOptions.OnlyOnFaulted);
Console.WriteLine(String.Format("Task {0} started", tasks[i].Id));
}
Console.ReadLine();
}
static void HandleException(Task task)
{
Console.WriteLine(String.Format("Task {0} stopped", task.Id));
// Check task status
for (int i = 0; i < tasks.Count(); i++)
{
Console.WriteLine(String.Format("Task {0} status = {1}", i, tasks[i].Status));
}
// fFind and restart the task here
if (tasks.Contains(task))
{
int index = Array.IndexOf(tasks, task);
tasks[index] = Task.Run(() => { Thread.Sleep(1000); throw new Exception("BOOM"); })
.ContinueWith(t => { HandleException(t); }
, TaskContinuationOptions.OnlyOnFaulted);
Console.WriteLine(String.Format("Task {0} started", tasks[index].Id));
}
}
我的应用程序会导致以下结果:
Task 3 started
Task 6 started
Task 5 stopped
Task 0 status = Running
Task 2 stopped
Task 1 status = Running
Task 0 status = Running
Task 1 status = Running
由于传递给HandleException(t)
的任务与原始任务不同,因此它在tasks[]
中未找到,因此未重新启动。同时查看tasks[]
的状态,它尚未停止。该如何正确地重新启动我的任务?
Task.Run
启动的,这会从线程池中获取一个线程 > 抛出异常 > 调用 HandleException,可能又在不同的线程上 > 再次调用Task.Run
并从线程池中获取另一个线程(不能使用相同的线程,因为它正在忙于执行 HandleException)> 当前线程完成执行并返回到池中。对于堆栈溢出,应该使用相同的线程递归执行函数,但这里我们有不同的线程(因此有不同的堆栈)。 - Evk