没有 pdb 文件的堆栈跟踪中更具体的异常位置

问题描述

在应用程序中,应始终尽最大努力确保代码经过全面测试,并以正确处理格式错误的输入或操作的方式编写。

虽然我们只是人,错误会蔓延,为了改善用户体验而不是毫不客气地崩溃到桌面,通常会捕获未处理的异常,向用户抛出消息并记录一些错误信息.

通常,PDB 不会随完成的程序一起分发,因此系统跟踪将不包含行号,这会导致难以找到引发错误的违规行。

以下面的示例程序为例(为简单起见,一个 .net 控制台应用程序)

using System;

namespace ExceptionTest
{
    class Program
    {
        static void Main(string[] args)
        {
            // Just in case we forget to catch something properly.
            System.AppDomain.CurrentDomain.UnhandledException += UnhandledExceptionTrapper;

            Console.WriteLine(DoStuff(10));

            Console.ReadKey();
        }

        static int DoStuff(int parameter)
        {
            int[] data = { 1,2,3,4,5,6 };

            int value = data[parameter]; // Improperly checked code

            return value;
        }

        static void UnhandledExceptionTrapper(object sender,UnhandledExceptionEventArgs e)
        {
            Console.WriteLine(e.ExceptionObject.ToString());
            Console.WriteLine("Press Enter to continue");
            Console.ReadLine();
            Environment.Exit(1);
        }
    }
}

如果它在没有 .pdb 文件的情况下运行,它将尽职尽责地捕获错误并报告它。

system.indexOutOfRangeException: Index was outside the bounds of the array.
   at ExceptionTest.Program.DoStuff(Int32 parameter)
   at ExceptionTest.Program.Main(String[] args)
Press Enter to continue

有什么办法可以在用户代码中包含发生错误的点,或者这只是不存在于可执行文件中(显然,我们可以看到 DoStuff 函数名称,其中包含了一些符号信息。

例如,除了堆栈跟踪之外,还有什么方法可以记录 int value = data[parameter];

解决方法

您在没有 PDB 的调用堆栈中看到的内容是可用的,因为您的 C# 代码是作为中间语言 (IL) 提供的。

这样想:你可以通过反射做的一切(例如,请求类和请求类的方法)在 IL 代码中也可用,因此可以成为堆栈跟踪的一部分。

但是,IL 代码仍然在发布版本中进行了优化,例如可以删除未使用的变量。

有什么办法可以包含这个点

没有。该信息位于 PDB 中。我们的想法是,您甚至可以为发布版本构建 PDB,并且至少为您提供给客户的版本store them in a symbol server

我也不只是捕捉异常并向用户显示信息或将其保存在日志文件中。相反,您可以使用您为此目的而存储的 PDB 在您的 PC 上分析 create a crash dump 文件 (.DMP)。

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...