如果失败则重新启动任务

MrE*_*een 5 c# continuations task

如果失败,我正在尝试重新启动多个任务之一。我使用.ContinueWith(t => { HandleException(t); }, TaskContinuationOptions.OnlyOnFaulted);HandleException(t)方法应该在现有任务数组中找到任务并在其位置创建一个新任务的位置。只有传递给我的HandleException(t)方法的任务与原始任务不同,所以我无法在我的现​​有任务中找到它。此外,在处理异常时,excisting 任务仍在运行。

例子:

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));
    }
}
Run Code Online (Sandbox Code Playgroud)

我的申请结果是:

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
Run Code Online (Sandbox Code Playgroud)

由于传递给 的HandleException(t)任务与原始任务不同,因此在 中找不到tasks[],因此不会重新启动。还看tasks[]它尚未停止的状态。如何正确重启我的任务?

Evk*_*Evk 4

这是因为您将延续任务存储在数组中,而不是原始任务(即 - 您存储 的结果ContinueWith)。要修复此问题,请存储原始任务:

var task = Task.Run(() => { Thread.Sleep(1000); throw new Exception("BOOM");});
task.ContinueWith(t => { HandleException(t); }
         , TaskContinuationOptions.OnlyOnFaulted);
tasks[i] = task;
Run Code Online (Sandbox Code Playgroud)

  • @MaximPodkolzin 不,因为所有事情都发生在不同的线程上。新任务通过“Task.Run”启动,这从线程池中获取一个线程&gt;抛出异常&gt;调用HandleException,可能再次在不同的线程上&gt;再次调用“Task.Run”并从线程池中获取另一个线程(无法使用相同的方法,因为它正忙于执行 HandleException) &gt; 当前线程完成执行并返回池。对于堆栈溢出,同一线程应该递归地执行函数,但这里我们有不同的线程(因此有不同的堆栈)。 (2认同)