在哪里处理Task抛出的异常

时间:2014-05-15 19:40:56

标签: c# .net multithreading task-parallel-library task

我正在单独的任务中执行一些轮询IO循环。这些循环可能会遇到异常。如果遇到异常,我想提醒调用者,使其可以:

  1. 记录
  2. 杀死所有IO线程
  3. 重置连接
  4. 重启IO线程
  5. 用户界面必须保持响应。处理此方案的首选方法是什么?我在下面列出了一个说明性的程序。

    using System;
    using System.Collections.Generic;
    using System.Linq;
    using System.Text;
    using System.Threading.Tasks;
    
    namespace TaskExceptionCatching
    {
        class Program
        {
            static void Main(string[] args)
            {
                startLoops();
    
                System.Console.WriteLine("Type 'Exit' when you're ready to stop.");
                while (System.Console.ReadLine() != "Exit")
                {
                    System.Console.WriteLine("Seriously, just type 'Exit' when you're ready to stop.");
                }
            }
    
            static private void startLoops()
            {
                System.Console.WriteLine("Starting fizzLoop.");
                var fizzTask = Task.Factory.StartNew(new Action(fizzLoop));
    
                System.Console.WriteLine("Starting buzzLoop.");
                var buzzTask = Task.Factory.StartNew(new Action(buzzLoop));
            }
    
            static private void fizzLoop()
            {
                while (true)
                {
                    //simulate something error prone, like some risky IO
                    System.Threading.Thread.Sleep(200);
                    bool isErr = (new Random().Next(1, 100) == 10);
                    if (isErr)
                        throw new Exception("Fizz got an exception.");
                }
            }
    
            static private void buzzLoop()
            {
                while (true)
                {
                    //simulate something error prone, like some risky IO
                    System.Threading.Thread.Sleep(200);
                    bool isErr = (new Random().Next(1, 100) == 10);
                    if (isErr)
                        throw new Exception("Buzz got an exception.");
                }
            }
        }
    }
    

2 个答案:

答案 0 :(得分:1)

这可能是async void方法很方便的极少数情况之一:

static async void StartAndMonitorAsync(Func<Task> taskFunc)
{
    while (true)
    {
        var task = taskFunc();
        try
        {
            await task;
            // process the result if needed
            return;
        }
        catch (Exception ex)
        {
            // log the error
            System.Console.WriteLine("Error: {0}, restarting...", ex.Message);
        }
        // do other stuff before restarting (if any)
    }
}

static private void startLoops()
{
    System.Console.WriteLine("Starting fizzLoop.");
    StartAndMonitorAsync(() => Task.Factory.StartNew(new Action(fizzLoop)));

    System.Console.WriteLine("Starting buzzLoop.");
    StartAndMonitorAsync(() => Task.Factory.StartNew(new Action(buzzLoop)));
}

如果您无法使用async/await,则可以使用Task.ContinueWith实现类似的逻辑。

如果可以多次调用startLoops(任务已经在#34;飞行中&#34;),则需要向StartAndMonitorAsync和任务添加取消逻辑它使用CancelltionToken"A pattern for self-cancelling and restarting task"中的更多详情)开始播放。

答案 1 :(得分:0)

来自MSDN &#34; Task基础结构将它们包装在AggregateException实例中。 AggregateException有一个InnerExceptions属性,可以枚举该属性以检查抛出的所有原始异常,并单独处理(或不处理)每个异常。即使只抛出一个异常,它仍然包含在AggregateException中。&#34; 有关详细信息,您可以check this

try
{
System.Console.WriteLine("Starting fizzLoop.");
            var fizzTask = Task.Factory.StartNew(new Action(fizzLoop));

            System.Console.WriteLine("Starting buzzLoop.");
            var buzzTask = Task.Factory.StartNew(new Action(buzzLoop));

}
catch (AggregateException ae)
{
    // Assume we know what's going on with this particular exception. 
    // Rethrow anything else. AggregateException.Handle provides 
    // another way to express this. See later example. 
    foreach (var e in ae.InnerExceptions)
    {
        if (e is MyCustomException)
        {
            Console.WriteLine(e.Message);
        }
        else
        {
            throw;
        }
    }

}