3

我无法理解 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 事件对象是帮助处理来自棘手的第三方库的异常的好方法。但是,异常似乎总是在任务中抛出。它看起来好像永远不会被TaskScheduler(或任何处理TaskExceptions的设施)捕获。

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

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

4

1 回答 1

2

这里(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

于 2012-06-03T22:13:15.113 回答