问题描述
在应用程序中,应始终尽最大努力确保代码经过全面测试,并以正确处理格式错误的输入或操作的方式编写。
虽然我们只是人,错误会蔓延,为了改善用户体验而不是毫不客气地崩溃到桌面,通常会捕获未处理的异常,向用户抛出消息并记录一些错误信息.
通常,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)。