51

作为我们产品中一些错误处理的一部分,我们想转储一些堆栈跟踪信息。但是,我们体验到许多用户只会截取错误消息对话框的屏幕截图,而不是向我们发送程序中可用的完整报告的副本,因此我想在此对话框中提供一些最小的堆栈跟踪信息。

我机器上的 .NET 堆栈跟踪如下所示:

at System.IO.__Error.WinIOError(Int32 errorCode, String maybeFullPath)
at System.IO.FileStream.Init(String path, FileMode mode, FileAccess access, Int32 rights, Boolean useRights, FileShare share, Int32 bufferSize, FileOptions options, SECURITY_ATTRIBUTES secAttrs, String msgPath, Boolean bFromProxy)
at System.IO.FileStream..ctor(String path, FileMode mode, FileAccess access, FileShare share, Int32 bufferSize, FileOptions options)
at System.IO.StreamReader..ctor(String path, Encoding encoding, Boolean detectEncodingFromByteOrderMarks, Int32 bufferSize)
at System.IO.StreamReader..ctor(String path)
at LVKWinFormsSandbox.MainForm.button1_Click(Object sender, EventArgs e) in C:\Dev\VS.NET\Gatsoft\LVKWinFormsSandbox\MainForm.cs:line 36

我有这个问题:

格式看起来是这样的:

at <class/method> [in file:line ##]

但是,atin关键字,我认为如果它们运行,例如,挪威 .NET 运行时而不是我安装的英文运行时,它们将被本地化。

有什么方法可以让我以与语言无关的方式分离这个堆栈跟踪,这样我就可以只显示那些条目的文件和行号?

换句话说,我想从上面的文字中得到这些信息:

C:\Dev\VS.NET\Gatsoft\LVKWinFormsSandbox\MainForm.cs:line 36

您可以提供的任何建议都会有所帮助。

4

5 回答 5

71

您应该能够通过说获得 StackTrace 对象而不是字符串

var trace = new System.Diagnostics.StackTrace(exception);

然后,您可以自己查看框架,而无需依赖框架的格式。

另请参阅:StackTrace 参考

于 2008-09-09T13:01:01.107 回答
28

这是我用来执行此操作的代码,没有例外

public static void LogStack()
{
  var trace = new System.Diagnostics.StackTrace();
  foreach (var frame in trace.GetFrames())
  {
    var method = frame.GetMethod();
    if (method.Name.Equals("LogStack")) continue;
    Log.Debug(string.Format("{0}::{1}", 
        method.ReflectedType != null ? method.ReflectedType.Name : string.Empty,
        method.Name));
  }
}
于 2009-08-06T14:08:48.937 回答
19

只是为了让这成为一个 15 秒的复制粘贴答案:

static public string StackTraceToString()
{
    StringBuilder sb = new StringBuilder(256);
    var frames = new System.Diagnostics.StackTrace().GetFrames();
    for (int i = 1; i < frames.Length; i++) /* Ignore current StackTraceToString method...*/
    {
        var currFrame = frames[i];
        var method = currFrame.GetMethod();
        sb.AppendLine(string.Format("{0}:{1}",                    
            method.ReflectedType != null ? method.ReflectedType.Name : string.Empty,
            method.Name));
    }
    return sb.ToString();
}

(基于林德霍尔姆的回答)

于 2011-05-22T13:26:59.303 回答
9

或者还有更短的版本..

Console.Write(exception.StackTrace);
于 2012-01-08T17:34:01.423 回答
1

作为替代方案,log4net 虽然有潜在危险,但比 System.Diagnostics 给了我更好的结果。基本上在 log4net 中,您有一个用于各种日志级别的方法,每个级别都有一个异常参数。因此,当您传递第二个异常时,它会将堆栈跟踪打印到您配置的任何附加程序。

例子:Logger.Error("Danger!!!", myException );

根据配置,输出看起来像

System.ApplicationException: Something went wrong.
   at Adapter.WriteToFile(OleDbCommand cmd) in C:\Adapter.vb:line 35
   at Adapter.GetDistributionDocument(Int32 id) in C:\Adapter.vb:line 181
   ...
于 2008-09-09T13:06:03.683 回答