【发布时间】:2011-12-14 13:06:13
【问题描述】:
这是什么意思以及如何解决?
我正在使用 TPL 任务。
整个错误
在等待任务或访问其异常属性时未观察到任务的异常。结果,未观察到的异常被终结器线程重新抛出。
在 System.Threading.Tasks.TaskExceptionHolder.Finalize()
mscorlib
【问题讨论】:
标签: c# wpf exception task task-parallel-library
这是什么意思以及如何解决?
我正在使用 TPL 任务。
整个错误
在等待任务或访问其异常属性时未观察到任务的异常。结果,未观察到的异常被终结器线程重新抛出。
在 System.Threading.Tasks.TaskExceptionHolder.Finalize()
mscorlib
【问题讨论】:
标签: c# wpf exception task task-parallel-library
如果您创建了一个任务,并且您从未调用task.Wait() 或尝试检索Task<T> 的结果,那么当垃圾收集器收集该任务时,它将在完成过程中关闭您的应用程序。有关详细信息,请参阅 Exception Handling in the TPL 上的 MSDN 页面。
这里最好的选择是“处理”异常。这可以通过延续来完成 - 您可以将延续附加到任务,并记录/吞下/等发生的异常。这提供了一种干净的方式来记录任务异常,并且可以写成一个简单的扩展方法,即:
public static void LogExceptions(this Task task)
{
task.ContinueWith( t =>
{
var aggException = t.Exception.Flatten();
foreach(var exception in aggException.InnerExceptions)
LogException(exception);
},
TaskContinuationOptions.OnlyOnFaulted);
}
通过上述方法,您可以防止任何任务通过以下方式关闭应用程序并记录它:
Task.Factory.StartNew( () =>
{
// Do your work...
}).LogExceptions();
或者,您可以订阅TaskScheduler.UnobservedTaskException 并在那里处理。
【讨论】:
Off,并将其用于您的全面延续。有助于消除因这个特殊例外而被压抑的一些挫败感。
.Net 4.0 是必需的。默认情况下,.net 4.5 中的异常处理已更改为不关闭应用程序。在Task Exception Handling in .NET 4.5中查看更多信息
当然;这意味着Task 在被留给垃圾回收后最终确定,但任务本身失败了。有两个修复:
ContinueWith(...)订阅,在参数中Task上勾选.IsFaulted和.Exception)TaskScheduler.UnobservedTaskException 事件,并将其标记为已观察(记录错误后调用e.SetObserved())【讨论】:
IsFaulted,您可以使用OnlyOnFaulted 延续选项并避免手动检查...
UnobservedTaskExceptionEventArgs 上的 SetObserved。