为什么订阅 Application.ThreadException 会吞噬异常

本文关键字:吞噬 异常 ThreadException Application 为什么 | 更新日期: 2023-09-27 18:32:15

假设我在消息循环中抛出了一个异常:

    private void timer1_Tick(object sender, EventArgs e)
    {
        throw new Exception("yehaaaa!!!!");
    }

默认情况下,这会抛出并向用户显示一般错误对话框。(这就是我想要的)

但是,如果我将以下订阅添加到Application.ThreadException:

    Application.ThreadException += new ThreadExceptionEventHandler(Application_ThreadException);
    private static void Application_ThreadException(object sender, ThreadExceptionEventArgs e)
    {
        //_raygunClient.Send(e.Exception);
    }

然后吞下异常。

为什么?

我怎样才能让它正常地扔给用户?

为什么订阅 Application.ThreadException 会吞噬异常

在参考源中都可以:

internal void OnThreadException(Exception t) {
    if (GetState(STATE_INTHREADEXCEPTION)) return;
    SetState(STATE_INTHREADEXCEPTION, true);
    try {
        if (threadExceptionHandler != null) {
            threadExceptionHandler(Thread.CurrentThread, new ThreadExceptionEventArgs(t));
        }
        else {
            if (SystemInformation.UserInteractive) {
                ThreadExceptionDialog td = new ThreadExceptionDialog(t);
如果有处理程序,

则调用该处理程序,否则将运行一些标准代码。如果要显示标准对话框,请使用 ThreadExceptionDialog 并以相同的方式处理 DialogResult。在我自己的代码中,有一些东西可以达到这种效果,这似乎有效:

private static void Application_ThreadException(object sender, ThreadExceptionEventArgs e)
{
    Exception exception = e.Exception;
    _Logger.Error(e.Exception, "An unhandled forms application exception occurred");
    // Show the same default dialog
    if (SystemInformation.UserInteractive)
    {
        using (ThreadExceptionDialog dialog = new ThreadExceptionDialog(exception))
        {
            if (dialog.ShowDialog() == DialogResult.Cancel)
                return;
        }
        Application.Exit();
        Environment.Exit(0);
    }
}