У меня возникли проблемы с пониманием того, как Исключения обрабатываются в 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 для печати на консоль и наблюдения за Исключением. Однако, как только выполнение достигнет результата задачи, из задачи будет выбрано WebException.