2

以下代码在 netcoreapp2.0 应用程序中运行时,似乎最终不会抛出UnobservedTaskException

using System;
using System.Threading;
using System.Threading.Tasks;

namespace ConsoleApp3 {

    class Program {

        public static void Main(string[] args) {

            TaskScheduler.UnobservedTaskException += (s, e) => {
                /// Try to crash the application - we wanna nail unobserved exceptions dead.
                /// Unfortunately, this code never seems to run.
                Console.WriteLine("UnobservedTaskException thrown.");
                throw e.Exception; 
            };

            var t = Task.Run(() => {
                throw new NotImplementedException();
            });

            while (!t.IsFaulted)
                Thread.Sleep(1);

            t = null;
            Console.WriteLine("Task is faulted.");

            GC.Collect();
            GC.WaitForPendingFinalizers();
            Console.ReadKey();
        }

    }
}

这是项目文件。我怎样才能让UnobservedTaskException处理程序被解雇?

<Project Sdk="Microsoft.NET.Sdk">

  <PropertyGroup>
    <OutputType>Exe</OutputType>
    <TargetFramework>netcoreapp2.2</TargetFramework>
    <LangVersion>7.3</LangVersion>
  </PropertyGroup>

</Project>

在其他 stackoverflow 帖子中,我看到了在项目文件中使用以下代码段的建议,但它仅适用于 .net 框架 4.0+ 项目。如果有 .netcore 项目文件的等效文件,那可能就是我需要知道的。

<runtime> 
    <ThrowUnobservedTaskExceptions enabled="true"/> 
</runtime> 
4

2 回答 2

4

我遇到过同样的问题。只需尝试在RELEASE模式下运行即可。我对其进行了测试,它与控制台应用程序 .net 核心版本 2.2 一起使用。

internal class Program
{
    private static void Main(string[] args)
    {
        // REMEMBER TO RUN IN RELEASE MODE

        var handler = new EventHandler<UnobservedTaskExceptionEventArgs>(Unobserved);
        TaskScheduler.UnobservedTaskException += handler;

        Task.Run(() => { Console.WriteLine("task 1"); throw new Exception("TASK 1 EXCEPTION"); });
        Task.Run(() => { Console.WriteLine("task 2"); throw new Exception("TASK 2 EXCEPTION"); });

        Thread.Sleep(1000);

        GC.Collect();
        GC.WaitForPendingFinalizers();
        GC.Collect();

        Thread.Sleep(1000);

        Console.ReadKey();
    }

    private static void Unobserved(object o, UnobservedTaskExceptionEventArgs e)
    {
        e.SetObserved(); // optional
        foreach (var ex in e.Exception.InnerExceptions)
        {
            Console.WriteLine(ex.Message);
        }
    }
}
于 2019-03-19T14:19:53.390 回答
1

还没有答案,但我觉得它有点有趣。我正在查看 GitHub 上的源代码。调用事件处理程序的唯一位置是类PublishUnobservedTaskException中的方法TaskScheduler

唯一PublishUnobservedTaskException被调用的地方是来自TaskExceptionHolder. PublishUnobservedTaskException但是,正如您在该源代码中看到的那样,在它调用之前必须满足一些条件。

我已启用调试 .NET 源代码,但我得到的版本与TaskExceptionHolderGitHub 上的不同,而且它显然与我尝试创建的断点不匹配。所以我不能确切地确认为什么它没有被调用,而且我今晚已经没有时间了。

甚至可能值得在 GitHub 上为此创建一个问题。

于 2018-12-22T04:50:24.907 回答