任务线程中抛出的异常,未被UnobservedTaskException捕获

时间:2012-06-03 21:27:03

标签: c# multithreading exception-handling task-parallel-library

我无法理解如何在TPL中处理异常。以下代码应说明我的问题。

using System;
using System.Collections.Generic;
using System.Net;
using System.Threading;
using System.Threading.Tasks;

namespace WebDLApp
{
    class Program
    {
        static void Main(string[] args)
        {
            TaskScheduler.UnobservedTaskException += TaskScheduler_UnobservedTaskException; // Not catching exception

            List<string> sites = new List<string>{ "http://microsoft.com", "http://yahoo.com", "http://facebook.com", "http://amazon.com", "http://foooo", "http://aol.com", "http://ask.com", "http://wikipedia.org" };
            List<Task<string>> tasks = new List<Task<string>>();


            foreach (string site in sites)
            {
                tasks.Add(Task.Factory.StartNew<string>((wsite) =>
                    {
                        using (WebClient wc = new WebClient())
                        {
                            wc.DownloadString((string)wsite); // Thrown here, always
                            return (string)wsite;
                        }
                    }, site)
                );
            }


            Task.WaitAll(tasks.ToArray()); // Can't catch here

            int counter = 1;
            foreach (var task in tasks)
            {

                Console.WriteLine(counter.ToString() + task.Result); // Can't catch here either
                counter++;
            }

            Console.ReadLine();
        }

        static void TaskScheduler_UnobservedTaskException(object sender, UnobservedTaskExceptionEventArgs e) // Never called
        {
            Console.WriteLine(e.Exception.Message);
            e.SetObserved();
        }
    }
}

据我所知,使用TaskScheduler.UnobservedTaskException事件对象是一种帮助处理棘手的第三方库异常的好方法。但是,异常似乎总是在Task中抛出。看起来它似乎永远不会被TaskScheduler(或任何设施处理TaskExceptions)捕获。

为了简洁起见,我省略了可能的try / catch位置,并用注释标记它们。

我期待TaskScheduler_UnobservedTaskException事件处理程序打印到控制台并观察异常。但是,一旦执行到达任务的结果,就会从任务中抛出WebException。

1 个答案:

答案 0 :(得分:2)

Here (How to handle exceptions with TaskScheduler.UnobservedTaskException?)就是解释。 只需替换

Task.WaitAll(tasks.ToArray()); // Can't catch here

Task.Factory.StartNew(() =>
{
    while (true)
    {
        Thread.Sleep(1000);
        GC.Collect();
    }
});
return;

查看TaskScheduler_UnobservedTaskException

中的消息