我无法理解 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。